[JZOJ6272] 2019.8.4【NOIP提高组A】整除
题目
题目大意
求方程\((x^m-x)\mod n=0\)在整数范围\([1,n]\)的解的个数。
\(n=\sum_{i=1}^{c}p_i\)
给出\(c\)和\(p_i\)
思考历程
作为数论白痴,比赛时看到这题就想要自闭了……
乱推一波式子,后面的就不会搞了。
于是就想部分分……结果连部分分都没有想出来。
直接打了个\(20\)分的暴力。
正解
可以把方程拆成\(c\)个方程(对于每个\(i\)):\((x^m-x)\mod p_i=0\)
分别解出每个同余方程组。解的时候枚举\(x\),并将每个\(x^m\)处理出来。
快速幂会TLE,所以要用积性筛的方式来求\(x^m\)。具体来说,函数\(f(x)=x^m\)是个积性函数。所以用快速幂求出\(x\)为质数时的\(x^m\),合数就是两个因数的函数值乘起来。
这样时间复杂度是可以过的。
处理出来之后,每个同余方程的解的个数的乘积就是整个同余方程组的解的个数。
证明;
对于方程\((x^m-x)\mod p_i=0\),设解为\(x_{i,1},x_{i,2},...,x_{i,s_i}\)
每个方程分别抽出一个解来,记作\(x_i\),\(x_i\)为\(x_{i,1..s_i}\)中的一个解。
设方程组的解为\(X\)
那么这个\(X\)满足以下方程组(对于每个\(i\)):
\(X \equiv x_i (\mod p_i)\)
根据中国剩余定理,这个方程组只会有一个解。
方程组的解和\(x_1,x_2,..,x_c\)的取值是一一对应的(这个可以感性理解)。
对于\(x_i\),有\(s_i\)种可能的取值。根据乘法原理,同余方程组解的个数即为每个同余方程的解的个数的乘积。
后来我知道了一个更加强大的方法。
同样是求\((x^m-x)\mod p_i=0\)的解的个数,然而这次不用暴力求。
有个性质:解的个数等于\(\gcd(m-1,p_i-1)+1\)
(为了方便,后面直接将\(p_i\)的下标省略。)
证明:
原式可以写成\(x^m\equiv x(\mod p)\)
\(p=2\)的时候显然成立。
\(x=0\)显然是方程的解
考虑解在区间\([1,p-1]\)的取值
由于\(p\)为奇素数,所以一定有原根。设原根为\(g\)
方程可以表示为\(g^{ym}\equiv g^y (\mod p)\)
由费马小定理得\(ym\equiv y (\mod (p-1))\)
也就是\(y(m-1) \equiv 0 (\mod (p-1))\)
设\(k=\gcd(m-1,p-1)\)。两边同时除以\(k\)得\(y\frac{m-1}{k}\equiv 0 (\mod \frac{p-1}{k})\)
由于\(\gcd(\frac{m-1}{k},\frac{p-1}{k})=1\),所以\(\frac{p-1}{k}|y\)
所以\(y\)为\(\frac{p-1}{k}\)的倍数,显然可以取\(0\)到\(k-1\)倍,也就是说\(y\)有\(k\)个解。
所以\(x\)也有\(k\)个解。
所以解的个数为\(\gcd(m-1,p_i-1)+1\)
有了这条性质,程序就快得飞起了(爆踩标程)……
代码
暴力求解:
(反正数据范围这么小,就懒得打线筛了。埃氏筛法的常数小一些)
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mo 998244353
#define P 10000
int id;
int c,m;
int n;
inline int my_pow(int x,int y,int p){
int res=1;
for (;y && res;y>>=1,x=x*x%p)
if (y&1)
res=res*x%p;
return res;
}
int di[P+1];
int xm[P+1];
inline void init(){
di[1]=1;
for (register int i=2;i<=P;++i){
if (di[i])
continue;
di[i]=i;
for (int j=i*i;j<=P;j+=i)
di[j]=i;
}
}
inline int work(int p){
int res=2;
xm[0]=0,xm[1]=1;
for (register int i=2;i<=p;++i){
xm[i]=(di[i]==i?my_pow(i,m,p):xm[i/di[i]]*xm[di[i]]%p);
res+=(xm[i]==i);
}
return res;
}
int main(){
freopen("division.in","r",stdin);
freopen("division.out","w",stdout);
int T;
scanf("%d%d",&id,&T);
init();
while (T--){
scanf("%d%d",&c,&m);
long long ans=1;
for (int i=1;i<=c;++i){
int p;
scanf("%d",&p);
ans=ans*work(p)%mo;
}
printf("%lld\n",ans);
}
return 0;
}
牛逼解法:
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mo 998244353
inline int gcd(int a,int b){
int k;
while (b){
k=a%b;
a=b;
b=k;
}
return a;
}
int main(){
freopen("division.in","r",stdin);
freopen("division.out","w",stdout);
int T;
scanf("%*d%d",&T);
while (T--){
int c,m;
scanf("%d%d",&c,&m);
long long ans=1;
while (c--){
int p;
scanf("%d",&p);
ans=ans*(gcd(p-1,m-1)+1)%mo;
}
printf("%lld\n",ans);
}
return 0;
}
总结
看来我的数论还是太菜了……QWQ……
[JZOJ6272] 2019.8.4【NOIP提高组A】整除的更多相关文章
- NOIP提高组2004 合并果子题解
NOIP提高组2004 合并果子题解 描述:在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消 ...
- 计蒜客 NOIP 提高组模拟竞赛第一试 补记
计蒜客 NOIP 提高组模拟竞赛第一试 补记 A. 广场车神 题目大意: 一个\(n\times m(n,m\le2000)\)的网格,初始时位于左下角的\((1,1)\)处,终点在右上角的\((n, ...
- 1043 方格取数 2000 noip 提高组
1043 方格取数 2000 noip 提高组 题目描述 Description 设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样 ...
- [NOIP提高组2018]货币系统
[TOC] 题目名称:货币系统 来源:2018年NOIP提高组 链接 博客链接 CSDN 洛谷博客 洛谷题解 题目链接 LibreOJ(2951) 洛谷(P5020) 大视野在线评测(1425) 题目 ...
- NOIP提高组初赛难题总结
NOIP提高组初赛难题总结 注:笔者开始写本文章时noip初赛新题型还未公布,故会含有一些比较老的内容,敬请谅解. 约定: 若无特殊说明,本文中未知数均为整数 [表达式] 表示:在表达式成立时它的值为 ...
- 津津的储蓄计划 NOIp提高组2004
这个题目当年困扰了我许久,现在来反思一下 本文为博客园ShyButHandsome的原创作品,转载请注明出处 右边有目录,方便快速浏览 题目描述 津津的零花钱一直都是自己管理.每个月的月初妈妈给津津\ ...
- 2018.12.30【NOIP提高组】模拟赛C组总结
2018.12.30[NOIP提高组]模拟赛C组总结 今天成功回归开始做比赛 感觉十分良(zhōng)好(chà). 统计数字(count.pas/c/cpp) 字符串的展开(expand.pas/c ...
- 2018.12.08【NOIP提高组】模拟B组总结(未完成)
2018.12.08[NOIP提高组]模拟B组总结 diyiti 保留道路 进化序列 B diyiti Description 给定n 根直的木棍,要从中选出6 根木棍,满足:能用这6 根木棍拼出一个 ...
- 2013 Noip提高组 Day2
3288积木大赛 正文 题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前 ...
- 2013 Noip提高组 Day1
3285 转圈游戏 2013年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 题目描述 Description ...
随机推荐
- 多个串的最长公共子串 SPOJ - LCS2 后缀自动机
题意: 求多个串的最长公共子串 这里用的是O(n)的后缀自动机写法 我后缀数组的专题有nlog(n)写法的 题解: 对于其中的一个串建立后缀自动机 然后对于后缀自动机上面的每一个节点求出每一个节点最长 ...
- 教你如何有效防止DDos攻击?
DDos又称分布式拒绝服务,DDos是利用大量合理的请求造成资源过载,导致服务不可用.就比如一个餐馆总共有100个座位,突然有一天某个商家恶意竞争,雇佣了200个人来到这个餐馆坐着不吃不喝,门口还排着 ...
- unicode_stop - 撤销控制台unicode模式(例如, 回到8-bit模式).
总览 unicode_stop 描述 unicode_stop 撤销以前 unicode_start(1) 命令的效果, 将显示屏和键盘设回到 8-bit 模式.
- JS对象 编程练习 某班的成绩出来了,现在老师要把班级的成绩打印出来。 效果图: XXXX年XX月X日 星期X--班级总分为:81
编程练习 某班的成绩出来了,现在老师要把班级的成绩打印出来. 效果图: XXXX年XX月X日 星期X--班级总分为:81 格式要求: 1.显示打印的日期. 格式为类似"XXXX年XX月XX日 ...
- where I will go
为什么我选择了java? 作为一个0基础入门的编程小白,大一的时候我接触了C语言,写了我的第一个程序hello world,内心有点小骄傲(我也能编程了),那时候还不知道java是什么,然而实际公司的 ...
- vue生态系统之vuex
一.webpack生成项目 1.webpack 在需要建立项目的目录中进行初始化项目 E:\vueProject>vue init webpack vuexpj ? Project name v ...
- Prometheus监控node-exporter常用指标含义
一.说明 最近使用Prometheus新搭建监控系统时候发现内存采集时centos6和centos7下内存监控指标采集计算公式不相同,最后采用统一计算方法并整理计算公式如下: 1 100-(node_ ...
- Java 并发总结(三)
锁优化及注意事项 有助于提高锁的性能 减小所持有时间:例如不要对方法直接加锁,而是在方法中对具体访问临界资源的代码加锁 减小锁粒度:如ConcurrentHashMap 用读写锁代替独占锁 锁分离:如 ...
- Golang 交叉编译跨平台的可执行程序 (Mac、Linux、Windows )
起因: 在项目中,我们每一次提交都需要添加commit 信息,而且我们的commit 信息,比较繁琐.我决定用golang语言编写一个小工具. 我决定使用语言:golang,使用工具:gox包. go ...
- shell 单引号&双引号的使用
使用双引号: shell> X='parameter' shell> echo "Hello $X" Hello parameter 单引号中嵌套单引号: shell& ...