zoukankan      html  css  js  c++  java
  • 洛谷P3702 [SDOI2017]序列计数

    题目大意:

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

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

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

    (100\%)的数据,(1leq n leq 10^9,1leq m leq 2 imes 10^7,1leq pleq 100)

    直接求不太好求,容斥一下,先求出全部的方案,再除掉没有质数的

    全部的方案怎么求?

    考虑(dp),设(f[i][j])表示(i)个数字,其和(mod p)(j)的方案数,可以得到转移方程(f[i_1+i_2][(j_1+j_2)\%p]=f[i_1][j_1]*f[i_2][j_2])

    然后跑一年就出来了

    考虑第一维,发现好像挺像个指数的运算

    那我们把第一维用快速幂优化掉

    当然我们要提前求出(i=1)时的(f[i][j]),这个循环一遍就完了

    (g[i][j])表示(i)个数字,其和(mod p)(j)的且不含质数方案数,转移方程相同,只是初始的时候质数不贡献答案

    然后就好了~

    #include<bits/stdc++.h>
    using namespace std;
    namespace red{
    #define int long long
    	inline int read()
    	{
    		int x=0;char ch,f=1;
    		for(ch=getchar();(ch<'0'||ch>'9')&&ch!='-';ch=getchar());
    		if(ch=='-') f=0,ch=getchar();
    		while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
    		return f?x:-x;
    	}
    	const int N=2e7+10,p=20170408;
    	int n,m,k;
    	int f[233],g[233],F[233],G[233],c[233];
    	signed prime[N>>1],num;
    	bool vis[N];
    	inline void work(int *a,int *b,int *d)
    	{
    		for(int i=0;i<k;++i)
    		{
    			for(int j=0;j<k;++j)
    			{
    				(c[i+j]+=a[i]*b[j])%=p;
    			}
    		}
    		for(int i=0;i<k;++i)
    		{
    			d[i]=(c[i]+c[i+k])%p;
    			c[i]=c[i+k]=0;
    		}
    	}
    	inline void main()
    	{
    		n=read(),m=read(),k=read();
    		f[1]=g[1]=F[0]=G[0]=1;
    		for(int i=2;i<=m;++i)
    		{
    			++f[i%k];
    			if(!vis[i]) prime[++num]=i;
    			else ++g[i%k];
    			for(int j=1;j<=num;++j)
    			{
    				if(i*prime[j]>m) break;
    				vis[i*prime[j]]=1;
    				if(i%prime[j]==0) break;
    			}
    		}
    		while(n)
    		{
    			if(n&1) work(F,f,F),work(G,g,G);
    			work(f,f,f);
    			work(g,g,g);
    			n>>=1;
    		}
    		printf("%lld
    ",(F[0]-G[0]+p)%p);
    	}
    }
    signed main()
    {
    	red::main();
    	return 0;
    }
    

    等等,我们发现了什么?

    看这里

    for(int i=0;i<k;++i)
    {
    	for(int j=0;j<k;++j)
    	{
    		(c[i+j]+=a[i]*b[j])%=p;
    	}
    }
    

    一个卷积!在这里写个任意模数(ntt)岂不美哉

    虽然对于这道题来说是没事找事

    代码先鸽子了,毕竟我还不会任意模数(ntt)

  • 相关阅读:
    C# Lambda表达式 (转)
    用C#读取txt文件的方法(转)
    c#中stringbuilder的使用(转)
    c# 日期和时间的获取(转)
    C# List<T>用法 泛型 (转)
    indent format codes
    格式化输入输出 小结
    putty connection manager 一些问题的整理
    linux 网络的一些书籍
    Oracle学习笔记
  • 原文地址:https://www.cnblogs.com/knife-rose/p/12052451.html
Copyright © 2011-2022 走看看