Alice想要得到一个长度为n的序列,序列中的数都是不超过m的正整数,而且这n个数的和是p的倍数。Alice还希望
,这n个数中,至少有一个数是质数。Alice想知道,有多少个序列满足她的要求。
标签:而且 alice nbsp highlight AC ini reg inpu 序列
BZOJ_4818_[Sdoi2017]序列计数_矩阵乘法
#include <stdio.h> #include <string.h> #include <algorithm> using namespace std; typedef long long ll; ll mod=20170408; int n,m,p,prime[7000050],cnt; bool vis[20000050]; struct Mat { ll v[105][105]; Mat() {memset(v,0,sizeof(v));} Mat operator*(const Mat a) const { Mat ans; int i,j,k; for(i=1;i<=p;i++) for(j=1;j<=p;j++) for(k=1;k<=p;k++) (ans.v[i][j]+=v[i][k]*a.v[k][j])%=mod; return ans; } }A,B; Mat pow(Mat x,int y) { Mat I; int i; for(i=1;i<=p;i++) I.v[i][i]=1; while(y) { if(y&1) I=I*x; x=x*x; y>>=1; } return I; } void init() { register int i,j; vis[1]=1; for(i=2;i<=m;i++) { if(!vis[i]) { prime[++cnt]=i; } for(j=1;j<=cnt&&i*prime[j]<=m;j++) { vis[i*prime[j]]=1; if(i%prime[j]==0) break; } } } int main() { int i,j; scanf("%d%d%d",&n,&m,&p); init(); for(i=1;i<=m;i++) { A.v[p][(i-1)%p+1]++; if(vis[i]) B.v[p][(i-1)%p+1]++; } for(i=p-1;i;i--) { for(j=1;j<=p;j++) { A.v[i][j]=A.v[i+1][j%p+1]; B.v[i][j]=B.v[i+1][j%p+1]; } } printf("%lld\n",(pow(A,n).v[p][p]-pow(B,n).v[p][p]+mod)%mod); }
标签:而且 alice nbsp highlight AC ini reg inpu 序列
原文地址:https://www.cnblogs.com/suika/p/8832021.html