题面

Alice想要得到一个长度为n的序列,序列中的数都是不超过m的正整数,而且这n个数的和是p的倍数。

Alice还希望,这n个数中,至少有一个数是质数。

Alice想知道,有多少个序列满足她的要求。

输入格式

一行三个数,n,m,p。

对100\%的数据,1<= n <=1e9,1<= m <= 2e7,1<= p <= 100

输出格式

一行一个数,满足Alice的要求的序列数量,答案对20170408取模。

题解

我很少写这样的矩阵快速幂的题解

由于p的范围很小,所以p^3*log是可以过的

我们先求不加质数限制的方案数,再减去所有数都为合数的方案。

这道题,其实我们要知道[1,m]中的这些数在p的剩余系中的分布就可以计算,

因为对于不加质数限制的方案,就算值域在[1 + p,m + p]它的答案也是正确的。

设f[i][j]表示对于i个数,“ 其总和 % p = j ”的方案数,

我们发现,对于f,余数是要相加的,而方案数是要相乘的,

根据这个来推一下矩阵,i = 1的情况要直接算:

然后答案矩阵 ,这里的n就是题目中的n,

再来推一下所有数都是合数的方案

其实只要用欧拉筛筛一遍合数就行,然后:

表示对于i个合数,“ 其总和 % p = j ”的方案数,

我们发现,对于f,余数是要相加的,而方案数是要相乘的,

根据这个来推一下矩阵,i = 1的情况要直接算:

然后答案矩阵 ,这里的n就是题目中的n,

最终的答案就是

别忘了要全程取模

CODE

#include<cstdio>
#include<iostream>
#include<cstring>
#include<vector>
#include<queue>
#include<stack>
#include<cmath>
#include<algorithm>
#define LL long long
#define MAXN 200005
#pragma GCC optimize(2)
#pragma G++ optimize(3)
#define rg register
#define DB double
using namespace std;
inline int read() {
int f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s == '-') f = -1;s = getchar();}
while(s >= '0' && s <= '9') {x = x * 10 + s - '0';s = getchar();}
return x * f;
}
int mod = 20170408;
int n,m,q,i,j,s,o,k,t;
struct matrix{
int n,m;
int a[105][105];
matrix(){memset(a,0,sizeof(a));n = m = 1;}
}m1;
matrix operator * (matrix A,matrix B) {
matrix C;
C.n = A.n;C.m = B.m;
for(rg int i = 1;i <= C.n;i ++) {
for(rg int k = 1;k <= A.m;k ++) {
for(rg int j = 1;j <= C.m;j ++) {
C.a[i][j] = (C.a[i][j] + A.a[i][k] *1ll* B.a[k][j] % mod) % mod;
}
}
}
return C;
}
matrix qkpow(matrix a,LL b) {
if(b == 0) return m1;
if(b == 1) return a;
matrix as = qkpow(a,b>>1);
return as*as*qkpow(a,b&1);
}
int pri[5000005],cn;
bool f[20000005];
void sieve(int n) {
f[1] = 1;cn = 0;
for(rg int i = 2;i <= n;i ++) {
if(!f[i]) pri[++cn] = i;
for(rg int j = 1;j <= cn && i * pri[j] <= n;j ++) {
f[i * pri[j]] = 1;
if(i % pri[j] == 0) break;
}
}
}
int cnt[105],cnt2[105];
int main() {
n = read();m = read();k = read();
for(rg int i = 1;i <= k;i ++) m1.a[i][i] = 1;
m1.n = m1.m = k;
sieve(m);
for(rg int i = 1;i <= m;i ++) {
if(f[i]) cnt2[i % k] ++;
cnt[i % k] ++;
}
matrix A,B,C,D;
A.n = 1,A.m = B.n = B.m = k;
for(rg int i = 1;i <= k;i ++) A.a[1][i] = cnt[i-1];
for(rg int i = 1;i <= k;i ++) {
for(rg int j = 1;j <= k;j ++) {
B.a[j][i] = cnt[(k + i - j) % k];
}
}
C = A * qkpow(B,n-1);
A.n = 1,A.m = B.n = B.m = k;
for(rg int i = 1;i <= k;i ++) A.a[1][i] = cnt2[i-1];
for(rg int i = 1;i <= k;i ++) {
for(rg int j = 1;j <= k;j ++) {
B.a[j][i] = cnt2[(k + i - j) % k];
}
}
D = A * qkpow(B,n-1);
printf("%d\n",(C.a[1][1] + mod - D.a[1][1]) % mod);
return 0;
}

最新文章

  1. 如何使用.NET开发全版本支持的Outlook插件产品(四)&mdash;&mdash;进阶探讨
  2. python &quot;yield&quot;(转载)
  3. [转]SpringMVC Controller&amp;View数据传递
  4. Java栈与堆一篇好文
  5. Jenkins-测试自动化(实例1-RF)
  6. android性能小贴士 翻译
  7. 怒刷DP之 HDU 1069
  8. 转载 ASP.NET MVC中使用ASP.NET Identity
  9. DOM操作--表格点击行变色
  10. IOS UIView 之属性篇
  11. jQuery 小知识点(插件)
  12. C++Primer笔记(1)
  13. 多线程 - pthread、NSThread
  14. ProtoBuf 与 gRPC
  15. Org mode无法生成LaTeX公式预览图片
  16. MySQL_视图
  17. 八卦一下Starlark语言
  18. Java并发编程:ThreadLocal的使用以及实现原理解析
  19. laravel migrate 指定执行部分 migration
  20. Excel如何固定表头,任意一行

热门文章

  1. 蓝牙、WiFi、ZigBee三大无线通信技术协议模块哪一个是最好的?
  2. 皓远的第二次博客作业(最新pta集,链表练习及期中考试总结)
  3. BUUCTF-穿越时空的思念
  4. FICO 常用事务码
  5. java程序使用ssl证书连接mysql
  6. tomcat JDK环境变量配置及tomcat多项目的配置
  7. centos7 离线升级/在线升级操作系统内核
  8. Python程序入口 __name__ == ‘__main__‘ 有重要功能(多线程)而非编程习惯
  9. 使用navicat连接远程linux mysql数据库出现10061
  10. HTML入门,基础知识