题目链接:

http://172.16.0.132/senior/#contest/show/2523/0

题目:

题解:(部分内容来自https://blog.csdn.net/gmh77/article/details/82947340)

首先我们容斥一下,设calc(l,r)为i∈[1,l],j∈[q,r]的方程的解的个数,显然答案等于calc(r2,r1)-calc(l1-1,r2)-calc(r1,l2-1)+calc(l1-1,l2-1)

考虑如何计算calc(l,r)

对于l和r,从低位向高位枚举每一个二进制位1,强制把这个1改成0,这样可以保证得到的数小于原来的数并且没有算重。假设l改变第i位,r改变第j位

(假设l不同的位比r后)

那么用红框表示已知部分,蓝框表示未知部分

所以异或之后就会变成这样

中间的紫色部分表示一半已知,一半未知,后面的蓝色部分表示完全未知


显然未知部分可以取到任何可能

考虑中间的紫色部分,由于$a$紫色部分确定,$b$紫色部分不确定,那么对于每一个$b$的紫色部分都对应一个$c$的紫色部分

也就是说,每一个$b$确定$2^{蓝色部分长度}$个$c$,且我们一共有$2^{max(i,j)}-1$个$b$。由于未知部分可以取到任何可能,所以我们一共有$2^{max(i,j)}-1$个$c$

考虑到每个c肯定是平等的,那么每个c就被计算了$\frac{2^{蓝色部分长度} \times (2^{max(i,j)}-1)}{2^{max(i,j)}-1}=2^{蓝色部分长度}$次

$mx=max(i,j)$,发现c的取值就是[((S/p[mx])*mx)^p[mx],((S/p[mx])*mx)^p[mx]+p[mx]-1],p[mx]=1<<mx

注意到第mx位是需要和原来相反的,所以要^p[mx]


还有三种情况

1.i==j

这个其实差不多,只是第mx位不需要取反,也就是不需要^p[mx]

2.a或b不改任何一位

这个也是一样的,注意一下变的那一个枚举的任何一位都需要取反

3.a和b都不变

这个直接异或一下直接判断就是了

#include<cstring>
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll; const int N=;
const int mod=;
ll m;
ll bin[N],p[N];
int a[N],b[N];
ll get(ll l,ll r)
{
if (l>r) return ;
if (l) return (r/m-(l-)/m)%mod;
else return (r/m+)%mod;//要考虑0
}
ll calc(ll l,ll r)
{
int A=-,B=-;
ll res=,S=l^r;
while (l)
{
a[++A]=l&;
l>>=;
}
while (r)
{
b[++B]=r&;
r>>=;
}
for (int i=;i<=A;i++)
if (a[i]) res=(res+get(((S/p[i])*p[i])^p[i],((S/p[i])*p[i])^p[i]+p[i]-))%mod;//b不变
for (int i=;i<=B;i++)
if (b[i]) res=(res+get(((S/p[i])*p[i])^p[i],((S/p[i])*p[i])^p[i]+p[i]-))%mod;//a不变
for (int i=;i<=A;i++)
if (a[i])
for (int j=;j<=B;j++)
if (b[j])
{
int mx=max(i,j);//特判i==j
if (i!=j) res=(res+get(((S/p[mx])*p[mx])^p[mx],((S/p[mx])*p[mx])^p[mx]+p[mx]-)*bin[min(i,j)])%mod;
else res=(res+get(((S/p[mx])*p[mx]),((S/p[mx])*p[mx])+p[mx]-)*bin[min(i,j)])%mod;
}
return res+((S%m)==);//i==l&&j==r
}
int main()
{
freopen("mod.in","r",stdin);
freopen("mod.out","w",stdout);
p[]=;bin[]=;
for (int i=;i<N;i++)
{
p[i]=p[i-]<<;
bin[i]=p[i]%mod;
}
ll l1,r1,l2,r2;
scanf("%lld%lld%lld%lld%lld",&l1,&r1,&l2,&r2,&m);
printf("%lld\n",((calc(r1,r2)-calc(l1-,r2)-calc(r1,l2-)+calc(l1-,l2-))%mod+mod)%mod);
return ;
}

最新文章

  1. 关于mysql 和Oracle的一大堆麻烦问题的解决方案
  2. Ubuntu Java Envrioment
  3. LoadRunner执行过程报错“Failed to connect to server &quot;xxx.xxx.xxx.xxx:xx&quot;:[10060] connetion time out”
  4. Rain on your Parade
  5. 20160805_CentOS6_键盘快捷键
  6. imageview圆角的实现
  7. css中的继承、层叠、样式优先级机制
  8. CentOS 6.4安装lnmp环境
  9. luigi学习9--执行模型
  10. Linux之父访谈录:设计内核只为了好玩
  11. com.intellij.javaee.oss.admin.jmx.JmxAdminException: com.intellij.execution.ExecutionException idea 导出war 报错
  12. 性能优化工具---top
  13. Ztree异步树加载
  14. ubuntu虚拟机和主机互ping及secureCRT使用
  15. p1154 地平线
  16. java 序列化 serialVersionUID 的作用 和 两种添加方式
  17. 计算几何细节梳理&amp;模板
  18. php持续推送信息到客户端的方法
  19. 力攻突破M20的5分钟BOLL
  20. topcoder srm 485 div1

热门文章

  1. Android 对ScrollView滚动监听,实现美团、大众点评的购买悬浮效果
  2. RAP开发入门-运行第一个HelloWorld(二)
  3. DWZ选项卡
  4. firefox工具
  5. 「JavaSE 重新出发」05.01 继承
  6. NSPort与NSRunloop的关系是流与消息调度的关系
  7. 【AnjularJS系列5 】— scopes、module、controller
  8. 页面定制CSS代码初探(一):页面变宽 文本自动换行 图片按比缩放
  9. CorelDRAW X6冰点价加推800套燃爆6月
  10. 教你用3ds max制作多边形小狗建模