Codeforces——1322B.Present

本文首发于我的blog,欢迎点击查看(无广告界面清爽!)

题目网址: https://codeforces.com/contest/1322/problem/B

写这篇博文是因为第一次遇到这个解法,对于我这个算法小白来说还是很新颖的。
PS.做题的时候天真的以为是O(n)的解法,并且可以用数学做。。。

题目

给定nn个数 a1a_1, a2a_2, … , ana_n
计算其两两之和的异或值。

分析

首先输入大小以看就不可以直接求(废话)
复杂度也就是压在O(nlogn)左右

关键点
我们可以把答案回到二进制,然后对二进制的每一位分开求。

原理

首先我们定义 xi=ak+ajx_i = a_k + a_j, 即 xix_i 是两数之和。
ss 是答案,sis_i 是指从右往左答案的第 ii 个二进制位(从0开始)

我们现在假设计算 sis_i,我们已知所有 xx 的第 ii 位的01情况,那么求 sis_i 就是对它们做异或,也就等于计算 xx 中第 ii 位是1的个数的奇偶情况。
即如果1有偶数个,那么答案对应的位数是0,反之则是1。(简单的异或运算原理)

Subproblem

对于结果ss,计算第 ii 位的值,即 sis_i

(等价问题)计算两数之和xx的第 ii 位是 11 的数量。

对于这个等价问题,我们先思考一个简单的情况:

假设现在有一个数 y<2i+1y < 2^{i+1},它的第ii位的是否位11取决于它是否在 [2i,2i+1)[2^{i}, 2^{i+1}) 内。
现在把 yy 扩大为正数,我们可以得到其第ii位的情况与 $y’ = y \% 2^{i+1} $。

因此,对于每个 xx,我们可以只考虑 $x \% 2^{i+1} $。

但我们希望能通过 aa 直接计算出个数,而非计算 xx 后求出解。
既然我们可以对 xx 取模,自然也可以对 aa 取模,这对加法并不会有影响。

因此,我们首先将 aa2i+12^{i+1} 取模,令取模后的数为 bb
我们可以得到 b[0,2i+1)b \in [0, 2^{i+1} )
即有 x[0,2i+21)x \in [0, 2^{i+2}-1 )

我们发现 xx 的范围缩小了很多,这次我们不对 xx 取模,
相反,我们可以直接对 xx 划出两个范围:[2i,2i+1)[2^i, 2^{i+1})[2i+2i+1,2i+21)[2^i + 2^{i+1}, 2^{i+2}-1 )

因此,我们希望寻找 xx 在这两个范围内的个数

Subsubproblem

对于结果 ss 的第 ii 位,现有 bb 是对 aa 对于 2i+12^{i+1} 取模,求 bb 两两之和在 [2i,2i+1)[2^i, 2^{i+1})[2i+2i+1,2i+21)[2^i + 2^{i+1}, 2^{i+2}-1 ) 的个数。

(简化版)给定数组 bb,求其两两和在某一区间的个数。

这里为简化,假设求的区间是 [k,t)[k, t)

暂时只有 O(nlogn)O(nlogn) 的解法,未知是否有更迅速的

首先将 bb 从小到大排序,
然后对 bib_i ,它可以相加的数为 bi+1b_{i+1}, …, bnb_n
用二分查找寻找下界值为 kbik-b_i 的下标 pp
用二分查找寻找上界值为 tbit-b_i 的下标 qq
满足的个数就是 qpq-p

代码

#include <bits/stdc++.h>

using namespace std;

long long a[400005], b[400005];

int main()
{
    ios::sync_with_stdio(false);
    long long n, i, j, k, t, m, p, q, s;
    cin >> n;
    m = 0;
    for (i = 0; i < n; i++) {
        cin >> a[i];
        m = max(a[i], m);
    }
    m = log2(m) + 2;
    s = 0;
    for (i = 0; i <= m; i++) {
        k = pow(2, i+1);
        for (j = 0; j < n; j++) {
            b[j] = a[j] % k;
        }
        sort(b, b+n);

        t = 0;
        for (j = 0; j < n; j++) {
            // 2^i, 2^(i+1)
            p = lower_bound(b+j+1, b+n, k/2 - b[j]) - b;
            q = lower_bound(b+j+1, b+n, k - b[j]) - b;
            t += q - p;

            // 2^i + 2^(i+1), 2^(i+2)-1
            p = lower_bound(b+j+1, b+n, k/2*3 - b[j]) - b;
            q = lower_bound(b+j+1, b+n, 2*k - 1) - b;
            t += q - p;
        }
        if (t % 2 == 1) {
            s += k / 2;
        }
    }
    cout << s << endl;
    return 0;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章