codeforces C. Vasily the Bear and Sequence 解题报告
题目链接:http://codeforces.com/problemset/problem/336/C
题目意思:给出一个递增的正整数序列 a1, a2, ..., an,要求从中选出一堆数b1, b2, ..., bk,这堆数进行完按位&运算之后(b1 and b2 and ... and bk),能被2v整除。除此,还要求所选中的这堆数是两两不同的,&之后要最大,换句话来说,v 要尽可能最大,如果找到最大的v的方法有多种,即在 a1, a2, ..., an 中有很多种不同的组合 & 完后能求得最大的v,则挑中的这堆数要尽可能最多。如果得到最多的数还是有多种,只需要写出其中一种即可,如果没找到符合条件的v,则输出-1。
这道题目想了4天多,看来要做出一条综合题的题目真的需要花费很长时间。一开始很天真的以为,暴力从a1往an搜,一个一个&,如果&完之后的结果是0,则置从&中最后运算完的数开始重新初始化。即假设给出序列1 2 3 4 5,1&2之后会等于0,于是不要&前的1,从2开始继续&,2&3......,直到最后一个数&完为止即可。很明显,这是不符合题意的,因为不能保证v最大。
(例如,给出10个数
109070199 215498062 361633800 406156967 452258663
530571268 670482660 704334662 841023955 967424642
错误的解法会得出 704334662 841023955 967424642, &的结果是
1,0000,0001,0000,1000,0000,1000,0010,v = 1。
而正确的结果是 361633800 406156967 452258663 530571268 841023955 967424642, &的结果是1,0000,0000,0000,0000,0000,0000,0000,v = 28。)
看来,&完之后的结果大并不意味着v会更大。关键取决于v的右边的0尽可能最多。
正确的思路:进行两轮筛选。考虑到 an 最大为109 ,且要使v最大,那么应该从v = 31(109 约为 2^30)开始遍历序列,分别尝试与序列中的每个数相&,如果不为0,代表这个数的第 v 位为1,把这个数先保存下来(第一轮筛选,并不保证是最后的结果), 接着从这些被筛选出的数中验证哪些符合第 v-1~1 位与一个第 v-1~1 位都为1的数(也就是下面所说的temp) 相&等于0,符合的就代表v是最优的,则为结果。题目的关键是设置一个临时变量temp,其从第1位到第v-1位都是1,每次将满足第v位为1的,与这个变量做&操作。
能得到这个思路,很谢谢hdu 群的silence,深感位运算的基础没打好啊,要继续努力才行。
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
using namespace std; const int maxn = + ;
int a[maxn], b[maxn], c[maxn]; // a[]是输入的序列,b[]是第一轮筛选出的数,c[]是第二轮即最后得出符合条件的数 int main()
{
int i, j, k, l, n, v, temp, flag;
while (scanf("%d", &n) != EOF)
{
for (i = ; i < n; i++)
{
scanf("%d", &a[i]);
}
for (v = ; v >= ; v--)
{
temp = ( << v) - ; // 关键所在,使得v-1~1位都置为1
// printf("tempout = %d\n", temp);
flag = j = ;
for (i = ; i < n; i++)
{
if (a[i] & ( << v)) // 挑出序列中满足第v位为1的数
{
// printf("a[%d] = %d\n", i, a[i]);
b[j++] = a[i];
// printf("b[%d] = %d\n", j-1, b[j-1]);
// printf("tempin1 = %d\n", temp);
temp &= a[i];
// printf("tempin2 = %d\n", temp);
}
}
if (j == )
continue; // 没找到进行下一轮循环
// printf("j = %d\n", j);
// printf("temp_num = %d\n", temp);
for (l = k = ; k < j; k++)
{
if (!(temp & b[k])) // 验证第一轮筛选出的数中是否符合第v-1~1位与temp中相&后是否都为0
{
temp &= b[k];
c[l++] = b[k];
// printf("c[%d] = %d\n", l-1, c[l-1]);
}
}
if (l == ) // l 保存找到的总个数
continue;
for (k = ; k < l; k++) // 找到一组解,且这个解满足是最优解
{
if (k == )
printf("%d\n%d", l, c[k]);
else
printf(" %d", c[k]);
}
printf("\n");
flag = ; // 找到符合条件的v的标志
break;
}
if (!flag) // 没有找到
printf("-1\n");
}
return ;
}
codeforces C. Vasily the Bear and Sequence 解题报告的更多相关文章
- codeforces A. Vasily the Bear and Triangle 解题报告
题目链接:http://codeforces.com/problemset/problem/336/A 好简单的一条数学题,是8月9日的.比赛中没有做出来,今天看,从pupil变成Newbie了,那个 ...
- codeforces 336C Vasily the Bear and Sequence(贪心)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud Vasily the Bear and Sequence Vasily the b ...
- C. Vasily the Bear and Sequence Codeforces 336C(枚举,思维)
C. Vasily the Bear and Sequence time limit per test 1 second memory limit per test 256 megabytes inp ...
- codeforces 336D Vasily the Bear and Beautiful Strings(组合数学)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud Vasily the Bear and Beautiful Strings Vas ...
- USACO Section2.1 Sorting a Three-Valued Sequence 解题报告
sort3解题报告 —— icedream61 博客园(转载请注明出处)---------------------------------------------------------------- ...
- codeforces B. Bear and Strings 解题报告
题目链接:http://codeforces.com/problemset/problem/385/B 题目意思:给定一条只有小写英文组成的序列,需要找出至少包含一个“bear”的单词的子序列个数.注 ...
- codeforces C. Cows and Sequence 解题报告
题目链接:http://codeforces.com/problemset/problem/284/C 题目意思:给出3种操作:t = 1:在前 a 个数中每个数都加上x: t= 2:在数组末尾增加一 ...
- codeforces 336D. Vasily the Bear and Beautiful Strings 组合数学 dp
题意: 给出n,m,g,求好串的个数 0 <= n,m <= 10^5,n + m >= 1,0 <= g <= 1 好串的定义: 1.只由0,1组成,并且恰好有n个0, ...
- timus 1175. Strange Sequence 解题报告
1.题目描述: 1175. Strange Sequence Time limit: 1.0 secondMemory limit: 2 MB You have been asked to disco ...
随机推荐
- 【bzoj1502】 NOI2005—月下柠檬树
http://www.lydsy.com/JudgeOnline/problem.php?id=1502 (题目链接) 今天考试题,从来没写过圆的面积之类的东西..GG 题意 一颗树由n个圆台组成,现 ...
- 洛谷P2327 [SCOI2005] 扫雷
题目描述 输入输出格式 输入格式: 第一行为N,第二行有N个数,依次为第二列的格子中的数.(1<= N <= 10000) 输出格式: 一个数,即第一列中雷的摆放方案数. 输入输出样例 输 ...
- Codevs2157 配对
题目描述 Description 给出2个序列A={a[1],a[2],…,a[n]},B={b[1],b[2],…,b[n]},从A.B中各选出n个元素进行一一配对(可以不按照原来在序列中的顺序), ...
- 如果您想省略JS里的分号,了解一下JS的分号插入原理吧
仅在}之前.一个或多个换行之后和程序输入的结尾被插入 也就是说你只能在一行.一个代码块和一段程序结束的地方省略分号. 也就是说你可以写如下代码 function square(x) { var n = ...
- 轻量级应用开发之(10) UINavigationController导航控制器
一 多控制器 1)一个iOS的app很少只由一个控制器组成,除非这个app极其简单2)当app中有多个控制器的时候,我们就需要对这些控制器进行管理3)有多个view时,可以用一个大的view去管理1个 ...
- 生成元(Digit Generator ,ACM/ICPC Seoul 2005 ,UVa 1583)
生成元:如果 x 加上 x 各个数字之和得到y,则说x是y的生成元. n(1<=n<=100000),求最小生成元,无解输出0. 例如:n=216 , 解是:198 198+1+9+8=2 ...
- Python socket编程之四:模拟分时图
建立 socket,先运行服务器,再运行客户端,建立连接后服务器从本地数据库调数据一截一截地发送给客户端,客户端接受数据绘图模拟分时图 1.socket # -*- coding: utf-8 -*- ...
- Unity3d三大光照渲染介绍
重要:在目前市面上常见的游戏引擎中,主要采用以下三种灯光实现方式: 顶点照明渲染路径细节 Vertex Lit Rendering Path Details 正向渲染路径细节 Forward Re ...
- C#线程 在某一时间内,只有N个线程在并发执行,其余都在队列中的实现(转载)
具体的需求是 在某一时间点,只有N个线程在并发执行,如果有多余的线程,则排队等候~ 还真是费尽心思啊~最终还是被我攻克了~ 下面我就来说说具体的实现 C#提供了Mutex与Interlocked这两个 ...
- visual studio 2010 破解版 破解方法
1.Microsoft Visual Studio 2010下载(均来自微软官网) 高级版(Premium) [建议下载] http://download.microsoft.com/do ...