O(1)求解自然数异或和
序
又是一个不眠之夜。
求:
\]
思路1:周期分析
\(O(logn)\)算法
考虑按位分析
对于\(f_i\)的第\(j\)位,它的值只与该位1出现次数有关。
而第\(j\)位1的出现又是呈周期性分布的。
我们考虑\(f_i=0 \bigoplus 1 \bigoplus 2 \bigoplus 3 \bigoplus...\bigoplus (i-1) \bigoplus i\)。
注意这里多加了一个0。
那么,在上式的各数中,第1位的变化为01010101
而第2位为00110011
第3位为00001111
以此类推。
周期分析
所以我们可以发现,第\(j\)位的值的出现是连续的,且每连续一组的相同值的个数为\(2^{j-1}\),这恰好是第\(j\)位的位权!
而对于数字的总个数,我们可以用\(x=i+1\)来表示。
分析第\(j\)位的值\((j \ge 2)\):
则第\(j\)位的出现的整组共有\(t=\lfloor{{x}\over{2^{j-1}}}\rfloor\)个,其中奇数组为0,偶数组为1,且其中出现数字1的总个数必定为偶数。
若\(t\)为奇数,说明剩余的不完整组的值为1,同时若\(x\)也为奇数,说明\(f_i\)的第\(j\)位为1;否则\(f_i\)的第\(j\)位为0。
由此,我们可以得到第\(j\)位的值\((j \ge 2)\)。
对于第1位,它出现的组共有\(x\)个,其中值为1的有\(\lfloor{{x}\over{2}}\rfloor\)个,故\(f_i\)的第1位等于\(x\)的第2位。
综上可以在\(O(logn)\)时间复杂度内求解。
\(O(1)\)算法
其实就是对\(O(logn)\)的算法作了一个小的总结。
分析第\(j\)位的值\((j \ge 2)\):
我们知道,当且仅当\(t = \lfloor{{x}\over{2^{j-1}}}\rfloor\)为奇数,同时\(x\)也为奇数时,第\(j\)位才为1;否则第\(j\)位为0。
体现在\(x\)这个数本身上,就是当\(x\)第1位为1时,\(f_i\)的第2位及以上与\(x\)的相同。
而当\(x\)第1位为0时,\(f_i\)的第2位及以上都为0。
然后第1位的特判很好处理,就是\(f_i\)的第1位等于\(x\)的第2位。
由此可以在\(O(1)\)时间复杂度内求解。
代码实现
闲来无事写个代码(因为太菜所以不会更简单的写法)
int xorsum(int x)
{
++x;
return ( (x&1) ? (x&(INT_MAX-1)) : 0 ) | ( (x&2) ? 1 : 0 );
}
数据检验
顺便学了一下二进制输出的方法
#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<bitset>
using namespace std;
int xorsum(int x)
{
++x;
return ( (x&1) ? (x&(INT_MAX-1)) : 0 ) | ( (x&2) ? 1 : 0 );
}
int main()
{
int n=10;
for(int i=1;i<=n;++i)
{
cout<<"number:"<<bitset<sizeof(i)*8>(i)<<'\n';
cout<<"xorsum:"<<bitset<sizeof(i)*8>(xorsum(i))<<'\n';
}
return 0;
}
输出如下:

思路2:数学化简
看了ltao思路和Limit给的正解,发现自己的做法实在是......太菜了。
一个很明显的劣势在于:我们的周期性分析是以\(x=i+1\)算出总数字个数再来分析性质的。
这样的转义分析最大的缺点就在于,性质推广,或者说移植的时候,会将定义转来转去,非常难以处理。
所以在这里再整理一下\(ltao\)的思路:
核心性质
定义\(g_{i,j}=i \bigoplus (i+1) \bigoplus ... \bigoplus j\)
有性质:\(g_{0,2^{k}-1} = g_{2^{k},2^{k+1}-1}(k \ge 1)\),即右式第\(k+1\)位全部被异或消掉
将左右两式异或可得:\(g_{0,2^{k+1}-1} = g_{0,2^{k}-1} \bigoplus g_{2^{k},2^{k+1}-1} = 0\)
得到核心性质:\(g_{0,2^{k-1}-1} = 0(k \ge 3)\),即可以用这个性质消掉函数中第\(k\)位为0的所有数,注意这个\(k\)的边界很重要!
\(O(logn)\)算法
有了这个性质,我们就可以很方便地对原式最高位进行化简,设所求函数\(g_{0,i}\)的\(i\)的最高位为\(k\),有:
\(g_{0,i} = g_{0, 2^{k-1}-1} \bigoplus g_{2^{k-1},i} = g_{2^{k-1},i}\)
然后组成\(g_{2^{k-1},i}\) 的数的第\(k\)位都为1,且共有\(m = i - 2^{k-1} + 1\)个这样的数
这样就可以化简掉第\(k\)位:
若\(i\)为奇数,则\(m\)为偶数,结果的第\(k\)位必然为0,即\(g_{2^{k-1},i} = g_{0,i-2^{k-1}}\);
若\(i\)为偶数,则\(m\)为奇数,结果的第\(k\)位必然为1,即\(g_{2^{k-1},i} = g_{0,i-2^{k-1}} + 2^{k-1}\)。
递归处理,至\(k \le 2\),达到了我们上面所说性质的边界以外,特判即可。
由此可以在\(O(logn)\)时间复杂度内求解。
\(O(1)\)算法
然后再对这个算法作一个小的总结:
我们可以发现上式的\(g_{0,i}-> g_{0,i-2^{k-1}}\)的过程当中,\(i\)的奇偶性始终不变。
因此只需一次分析起始状态\(g_{0,i}\):
若\(i\)为奇数,则\(m\)为偶数,结果的第3位及以上都为0。
若\(i\)为偶数,则\(m\)为奇数,结果的第3位及以上与\(i\)的相同。
剩下两位特判即可。
由此可在\(O(1)\)时间复杂度内完成求解。
代码实现
int f[4]={0,1,3,0};
int xorsum(int x)
{
return ( (x&1) ? 0 : (x&(INT_MAX-3)) ) | f[x&3];
}
或者更直观的写法:
int xorsum(int x)
{
switch(x&3)
{
case 0:
return x;
case 1:
return 1;
case 2:
return x+1;
case 3:
return 0;
}
}
非常重要的一点在于,这种直接将x&3的值和xorsum的值对应的直观函数表示,能够有效地解决一些扩展问题,有兴趣的可见这道基于Limit's idea的水题。
O(1)求解自然数异或和的更多相关文章
- [笔记] 整除分块 & 异或性质
整除分块 参考资料:整除分块_peng-ym OI生涯中的各种数论算法的证明 公式 求:\(\sum_{i=1}^{n}\lfloor\frac{n}{i}\rfloor\) 对于每个\(\lfloo ...
- 【bzoj4671】异或图(容斥+斯特林反演+线性基)
传送门 题意: 给出\(s,s\leq 60\)张图,每张图都有\(n,n\leq 10\)个点. 现在问有多少个图的子集,满足这些图的边"异或"起来后,这张图为连通图. 思路: ...
- poj1222(高斯消元法解异或方程组+开关问题)
题目链接:https://vjudge.net/problem/POJ-1222 题意:给定一个5×6的01矩阵,改变一个点的状态时它上下左右包括它自己的状态都会翻转,因为翻转2次等价与没有翻转,那么 ...
- [总结]其他杂项数学相关(定理&证明&板子)
目录 写在前面 一类反演问题 莫比乌斯反演 快速莫比乌斯变换(反演)与子集卷积 莫比乌斯变换(反演) 子集卷积 二项式反演 内容 证明 应用举例 另一形式 斯特林反演 第一类斯特林数 第二类斯特林数 ...
- (转载)solr时区问题解决方案
solr默认的使用的是utc格林尼治时间,与我们的GMT+8相差8个小时,网上好多解决办法是在自己应用中的时间上加8个小时和减8个小时做变换:或者不用date类型,改为long. 个人感觉这两个办法都 ...
- CodeForces 703D Mishka and Interesting sum
异或运算性质,离线操作,区间求异或和. 直接求区间出现偶数次数的异或和并不好算,需要计算反面. 首先,很容易求解区间异或和,记为$P$. 例如下面这个序列,$P = A[1]xorA[2]xorA[3 ...
- [AHOI2001]质数和分解
[AHOI2001]质数和分解 题目描述 任何大于 1 的自然数 n 都可以写成若干个大于等于 2 且小于等于 n 的质数之和表达式(包括只有一个数构成的和表达式的情况),并且可能有不止一种质数和的形 ...
- 洛谷 P2563 [AHOI2001]质数和分解
洛谷 P2563 [AHOI2001]质数和分解 题目描述 任何大于 1 的自然数 n 都可以写成若干个大于等于 2 且小于等于 n 的质数之和表达式(包括只有一个数构成的和表达式的情况),并且可能 ...
- 【省选水题集Day1】一起来AK水题吧! 题目(更新到B)
题解:http://www.cnblogs.com/ljc20020730/p/6937954.html 水题A: [AHOI2001]质数和分解 题目网址: https://www.luogu.or ...
随机推荐
- [原创][开源] SunnyUI.Net 字体图标
SunnyUI.Net, 基于 C# .Net WinForm 开源控件库.工具类库.扩展类库.多页面开发框架 Blog: https://www.cnblogs.com/yhuse Gitee: h ...
- JDK Base64编解码1.7和1.8的坑
场景 对接一个第三方api接口,其中签名部分用的是JDK8的编码.我们线上采用JDK7,导致项目无法编译 替换编解码部分为1.7的代码,然后签名又不对 所以坑就在这里,结论,1.7的编解码有换行符导致 ...
- hadoop知识整理(2)之MapReduce
之前写的关于MR的文章的前半部分已丢. 所以下面重点从3个部分来谈MR: 1)Job任务执行过程,以及主要进程-ResourceManager和NodeManager作用: 2)shuffle过程: ...
- [noi.ac省选模拟赛]第12场题解集合
题目 比赛界面. T1 数据范围明示直接\(O(n^2)\)计算,问题就在如何快速计算. 树上路径统计通常会用到差分方法.这里有两棵树,因此我们可以做"差分套差分",在 A 树上对 ...
- [NOI Online #3]魔法值
题目 点这里看题目. 分析 我们不难想到,对于系数进行一下的拆分: \[\begin{aligned} f(u,j)&=\bigoplus_{(u,v)\in E} f(v,j-1)\ ...
- 一张图搞懂Ubuntu安装时姓名、计算机名、用户名
安装Ubuntu时会要求填写如下图的信息: 感谢:苏守坤 注意:上面的博客讲述了各自的具体含义,本篇博客只是说明这些名称在系统安装后会出现的位置.
- 【Vulnhub】FristiLeaks v1.3
靶机信息 下载连接 https://download.vulnhub.com/fristileaks/FristiLeaks_1.3.ova.torrent https://download.vuln ...
- C++_继承
C++支持单继承和多继承,并提供3类继承:public.private及protected.在public继承中,每个派生类对象都是基类对象,反之基类对象却不是派生类对象.派生类成员无法直接访问基本的 ...
- 微信小程序-返回并更新上一页面的数据
小程序开发过程中经常有这种需求,需要把当前页面数据传递给上一个页面,但是wx.navigateBack()无法传递数据. 一般的办法是把当前页面数据放入本地缓存,上一个页面再从缓存中取出. 除此之外还 ...
- leetcode 6 z字型变换
执行用时 :64 ms, 在所有 Python3 提交中击败了99.74%的用户由题目可知 我们的最终字符串会被摆成 numRows 行,那我们理解为 最终结果是numRows个字符串相加 先建立等于 ...