牛客网提高组模拟赛第五场 T1同余方程(异或)(位运算)



区间不好做,但是我们可以转化成前缀来做。转化为前缀之后之后就是二维前缀和。
但是我还是不怎么会做。所以只能去看吉老师的题解 (确定写的那么简单真的是题解???)。
我们要求模一个数余0,就等于找它的倍数。找它的倍数自然只要知道区间就可以了。
题解上面说:如果\(r1=2^n\),\(r2=2^m\),不妨设 \(n<=m\),那么结果一定在区间 \([0,2^n)\)中,而 且每一个值出现了 \(2^m\) 次
这是很显然的,因为按照异或的运算法则来讲,不可能会有比r1最高位1还高的位出现。而且考虑在二进制下,还有多种不同的方式可以凑成这个数。
之后我们考虑更一般的情况。
我们把高位确定下来,之后摆放低位。两两之间统计答案。
怎么理解呢,就是先把最高位的1确定下来,然后后面显然1就可以随便放了(反正使得当前位1变成0,后面(比它小的)的所有数都可以达到了)。我们考虑遍历枚举的两个边界的最高位1的位数,然后计算这种组合能够达到的范围(啊啊啊语文能力又下线了。。。怎么办。。大家看一下代码???),知道范围之后就容易能够知道有多少个所求数的倍数了。之后按照上面解释的,我们可以统计出有多少种不同的x,y组合方式达到这个异或值。
因为感觉自己说的很不清楚,所以代码里加了一点注释qwq。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#define mod 998244353
#define ll long long
using namespace std;
ll kkk,ans,a,b,c,d;
ll calc(ll a,ll b){
ll ansans=0;
for(int i=0;i<=63;++i)
{
//因为在long long范围内,所以遍历到63
if(((a>>i)&1)==0) continue;
//如果该位不是1就跳过
for(int j=0;j<=63;++j)
{
if(((b>>j)&1)==0) continue;
//同上
ll l=((a-(1ll<<i))^(b-(1ll<<j)))&(~((1ll<<max(i,j))-1));
//因为我们已经确定max(i,j)为最高位1了
//所以前面的值就不变了,直接异或就行
//后面的max(i,j)位经过处理全变成0(与操作和后面的东西就是这个用)
//&(~((1<<n)-1))表示将后面的n位全部变成0
ll r=l+(1ll<<max(i,j))-1;
//因为是左闭右开区间
ll cnt=(r/kkk)-l/kkk+(l%kkk==0);
//注意左闭,所以如果正好是倍数的话还是要加上1
tmp%=mod;
ansans=(ansans+(cnt*((1LL<<min(i,j))%mod))%mod)%mod;
//注意及时取模qwq
//其实像我这样写不好,最好分步写,避免中间没有取模爆long long
//哦,还要注意左移右移的运算级很低,所以即使加括号
}
}
return ansans;
}
int main()
{
freopen("ce.in","r",stdin);
scanf("%lld %lld %lld %lld %lld",&a,&b,&c,&d,&kkk);
ll ans=calc(b+1,d+1);
ans=(ans+mod-calc(c,b+1))%mod;
ans=(ans+mod-calc(a,d+1))%mod;
ans=(ans+calc(a,c))%mod;
//类比二维前缀和
printf("%lld\n",ans%mod);
return 0;
}
牛客网提高组模拟赛第五场 T1同余方程(异或)(位运算)的更多相关文章
- nowcoder(牛客网)提高组模拟赛第四场 解题报告
T1 动态点分治 就是模拟..... 但是没有过!! 看了题解之后发现.... 坑点:有可能 \(x<=r\),但是
- 牛客网提高组模拟赛第七场 T3 洞穴(附bitset介绍)
就是DP. 我们可以很简单的想到要枚举中间点,进行边数的转移. 但是因为边长数据范围很大,所以我们考虑log的倍增. 状态设计为\(dp[i][j][k]\),为从节点\(i\)走\(2^k\)步能否 ...
- 牛客网提高组模拟赛第七场 T2 随机生成树
其实看懂题就很水啦qwq,就是求\(1-N\)的约数啦. 暴力求的话时间复杂度是\(O(NlogN)\)的,其实正解是枚举每个数的倍数......这样的时间复杂度是\(\frac{N}{1}+\fra ...
- nowcoder(牛客网)提高组模拟赛第一场 解题报告
T1 中位数(二分) 这个题是一个二分(听说是上周atcoder beginner contest的D题???) 我们可以开一个数组b存a,sort然后二分b进行check(从后往前直接遍历check ...
- nowcoder(牛客网)普及组模拟赛第一场 解题报告
蒟蒻我可能考了一场假试 T1 绩点 这题没什么好说的,应该是只要会语言的就会做. T2 巨大的棋盘 一个模拟题吧qwq,但是要注意取模的时候先加上n或者m再取模,要不然会错的. #include< ...
- 计蒜客NOIP2017提高组模拟赛(五)day1-展览
传送门 发现这题选或不选对状态的优劣程度不会产生影响,如果已经确定了两个数a和b,那么最优的首项和公比也都是唯一确定的, 与对于后面的数x,加进去也好不加进去也好,首项和公比依旧是原来的 于是我们用尺 ...
- 计蒜客NOIP2017提高组模拟赛(五)day1-机智的 AmyZhi
传送门 很水的题目啦QAQ #include<cstdio> #include<cstdlib> #include<algorithm> #include<c ...
- 计蒜客NOIP2017提高组模拟赛(五)day2-蚂蚁搬家
传送门 这题可以用线段树来维护 #include<cstdio> #include<cstdlib> #include<algorithm> #include< ...
- 计蒜客NOIP2017提高组模拟赛(五)day2-成绩统计
传送门 用hash,因为map的复杂度可能在这题中因为多一个log卡掉,但是hash不会 可能因为这个生成的随机数有循环的情况,不是完全均匀的 而且这题hash表的长度也可以开的很大 #include ...
随机推荐
- Scanner和BufferReader之区别
在Java SE6中我们可知道一个非常方便的输入数据的类Scanner,位于java.util包中,这个Scanner的具体用法为Scanner in = new Scanner(System.in) ...
- uml 时序图
1.时序图的概念 时序图定义 : 描述了对象之间传递消息的时间顺序, 用来表示用例中的行为顺序, 是强调消息时间顺序的交互图; 时序图描述的事物: 时序图描述系统中类和类之间的交互, 将这些交互建模成 ...
- [转]字符集、字符编码、XML中的中文编码
字符集.字符编码.XML中的中文编码 作为程序员的你是不是对于ASCII .UNICODE.GB2321.UTF-7.UTF-8等等不时出现在你面前的这些有着奇怪意义的词感到很讨厌呢,是不是总觉得好象 ...
- Java-集合条件筛选
import java.lang.reflect.Field; import java.lang.reflect.Modifier; import java.util.ArrayList; impor ...
- faster-rcnn训练自己的数据集参考文章
https://www.cnblogs.com/CarryPotMan/p/5390336.html
- 606. Construct String from Binary Tree 从二叉树中构建字符串
[抄题]: You need to construct a string consists of parenthesis and integers from a binary tree with th ...
- 第一个Django应用程序_part2
一.数据库配置 此文延续第一个Django应用程序_part1. 打开mystic/settings.py.这是一个普通的Python模块,其模块变量表示Django配置 默认情况下,配置使用SQLi ...
- Zookeeper使用--开源客户端
一.ZkClient ZkClient是在Zookeeper原生API接口之上进行了包装,是一个更易用的Zookeeper客户端,其内部还实现了诸如Session超时重连.Watcher反复注册等功能 ...
- STL中mem_fun, mem_fun_ref用法
1.引言 先看一个STL中for_each的用法: #include <iostream> #include <vector> #include <algorithm&g ...
- ubuntu如何安装samba
1.samba安装sudo apt-get install samba2.修改smb.confsudo gedit /etc/samba/smb.conf 文件最后增加如下代码:[share] pat ...