FJNU 1153 Fat Brother And XOR(胖哥与异或)

FJNU 1153 Fat Brother And XOR(胖哥与异或)

Time Limit: 1000MS   Memory Limit: 257792K

【Description】

【题目描述】

Fat brother had master ACM, recently he began to study the operation of XOR (the operation “^”). He thought of a very interesting question: select arbitrary k positive integers from the n positive integers, then XOR the selected k digital, sum all cases of XOR. Now he wants to quickly calculate the results. Maybe the results will be great, just modulo 20162333.

 

For example, 3 integers: 1 2 3, select arbitrary 2 positive integers, the all cases:

 

{1,2}, {2, 3}, {1, 3}. So the results is {(1 ^ 2) + (2 ^ 3) + (1 ^ 3)} %20162333

胖哥是个ACM大牛,最近他在学习异或操作(运算符为“^”)。他想到了个有趣的问题:从n个正整数中任选k个,然后对这k个数异或,对所有异或的结果求和。现在他想快速计算结果。最终结果可能很大,于是模20162333。

 

例如,3个整数:1 2 3,选择任意2个正整数,所有情况为:

 

{1,2}, {2, 3}, {1, 3}。因此结果为{(1 ^ 2) + (2 ^ 3) + (1 ^ 3)} %20162333

 

【Input】

【输入】

There are multiple test cases. The first line of input contains an integer T (T <= 20) indicating the number of test cases. For each test case:

 

The first line contains two integer n, k (1 <= k <= n <= 1000)

 

The second line contains n integer ai (1 <= ai <= 1000000000)

多组测试用例。

第一行是一个整数T(T <= 20)表示测试用例的数量。对于每个测试用例:

 

第一行有两个整数,k(1 <= k <= n <= 1000)

 

第二行有n个整数ai(1 <= ai <= 1000000000)

 

 

【Output】

【输出】

For each test case, output the sum % 20162333

对于每个测试用例,输出和%20162333

 

【Sample Input - 输入样例】

【Sample Output - 输出样例】

2

3 2

1 2 3

6 2

6 6 6 6 6 6

6

0

 

【Hint】

【提示】

The

first sample test case: (1 ^ 2) + (2 ^ 3) + (1 ^ 3) = 6

 

 

The

second sample test case: (6 ^ 6) * 15 = 0

第一个样例:

(1 ^ 2) + (2 ^ 3) + (1 ^ 3) = 6

 

第二个样例:

(6 ^ 6) * 15 = 0

 

【题解】

这道题需要以二进制的观点来看待每个ai

举个例子:

Xai表示二进制的下ai的第X位(为方便表示从0开始)

(a1 ^ a2) + (a2 ^ a3) + (a1 ^ a3)

=[(0a1 ^ 0a2) + (0a2 ^ 0a3) + (0a1 ^ 0a3)]*20 +

[(1a1 ^ 1a2) + (1a2 ^ 1a3) + (1a1 ^ 1a3)]*21 +

[(2a1 ^ 2a2) + (2a2 ^ 2a3) + (2a1 ^ 2a3)]*22 +

…………………………………………………………………………… +

[(Xa1 ^ Xa2) + (Xa2 ^ Xa3) + (Xa1 ^ Xa3)]*2X

 

此时只剩0和1了

我们可以很轻易地知道:0 ^ 0 = 0, 1 ^ 0 = 1, 1 ^ 1 = 0

当然了,a ^ b = b ^ a

所以除了1 ^ 0之外,其他结果为0的情况都是无用的。

 

因此,题目转换为:求二进制下,1 ^ 0 这种情况出现的次数。

 

统计ai中二进制第X位出现几次1,保存到bit[x]

 

第X位提供的1 = 2X * 有效情况数

有效情况数就是由bit[X]个1与 n-bit[X]个0组合出来的

并且只有取奇数个1的时候,异或的结果才能为1(1 ^ 1 = 0)

由此可以得出

当前有效情况数: 

最后利用递推公式快速求组合数c(n,m)=c(n-1,m-1)+c(n-1,m)

(当然你把这个公式当成杨辉三角也是可以的:))

【代码 C++】

 1 #include<cstdio>
 2 #include <cstring>
 3 #define  mx 1005
 4 #define mod 20162333
 5 int main(){
 6     int t, n, k, i, j, w, opt, bit[30], c[mx][mx];
 7     for (i = 0; i < mx; ++i){
 8         for (c[i][0] = c[i][i] = j = 1; j < i; ++j)
 9             c[i][j] = (c[i - 1][j] + c[i - 1][j - 1]) % mod;
10     }
11     while (~scanf("%d", &t)){
12         while (t--){
13             memset(bit, 0, sizeof(bit));
14             scanf("%d%d", &n, &k);
15             for (i = 0; i < n; ++i){
16                 scanf("%d", &j);
17                 for (w = 0; j; j >>= 1) bit[w++] += j & 1;
18             }
19             for (i = opt = 0; i < 30; ++i){
20                 for (j = 1; j <= k; j += 2){
21                     opt += (1LL << i) *c[bit[i]][j] % mod*c[n - bit[i]][k - j] % mod;
22                     opt %= mod;
23                 }
24             }
25             printf("%d\n", opt);
26         }
27     }
28     return 0;
29 }
 1 #include<cstdio>
 2 #include <cstring>
 3 #define  mx 1005
 4 #define mod 20162333
 5 int read_int(){
 6     int add = getchar() - '0';
 7     int a = getchar();
 8     while (a >= '0' && a <= '9') add = add * 10 + a - '0', a = getchar();
 9     return add;
10 }
11 int main(){
12     int t, n, k, i, j, w, opt, bit[30], *c[mx];
13     for (i = 0; i < mx; ++i){
14         c[i] = new int[i + 1];
15         for (c[i][0] = c[i][i] = j = 1; j < i; ++j)
16             c[i][j] = (c[i - 1][j] + c[i - 1][j - 1]) % mod;
17     }
18     while (t = read_int(), t>0){
19         while (t--){
20             memset(bit, 0, sizeof(bit));
21             n = read_int(); k = read_int();
22             for (i = 0; i < n; ++i){
23                 for (w = 0, j = read_int(); j; j >>= 1) bit[w++] += j & 1;
24             }
25             for (i = opt = 0; i < 30; ++i){
26                 for (j = 1; j <= k; j += 2){
27                     if (bit[i] < j || n - bit[i] < k - j) continue;
28                     opt += (1LL << i) *c[bit[i]][j] % mod*c[n - bit[i]][k - j] % mod;
29                     opt %= mod;
30                 }
31             }
32             printf("%d\n", opt);
33         }
34     }
35     return 0;
36 }
伪·优化


 

posted @ 2016-04-24 00:52  Simon_X  阅读(466)  评论(0编辑  收藏  举报