原文链接https://www.cnblogs.com/zhouzhendong/p/SRM502-1000.html

SRM502 Div1 1000

题意

从 [0,n-1] 中选择 k 个不同的整数,使得他们的和是 n 的倍数,求方案数。对 \(10^9+7\) 取模。

\(n\leq 10^9,k\leq 1000\)

题解

​ 首先我们考虑从 n 个里面选择 k 个并进行排列的方案数,最终只需要除以 k! 就好了。

​ 设 \(M=n\) ;

​ 设 \(f(n,m,t)\) 表示 在 \(0,1,\cdots M-1\) 中任选 \(n\) 个,并使得 \((\sum_{1\leq i < n } x_i ) + t x_n \equiv 0 \pmod m\cdots (1)\) (设取的第 \(i\) 个为 \(x_i\) ) 。

​ 则答案显然是 \(f(k,n,1)\) 。

​ 于是我们考虑如何求解这个函数。

\(f(n,m,t)=\)

  1. 如果 $n=0 $ ,那么返回 \(1\) 。

  2. 如果 \(m = 1\) ,那么 (1) 式恒成立,答案就是 \(n!\binom{M}{n} = M^{\underline{n}}\)

  3. 否则,我们通过容斥,分类讨论(结果就是下面 (1) 的贡献减掉 (2) 的贡献)。

    (1) \(x_i(i<n)\) 中可能存在与 \(x_n\) 相等的数(也可能不存在):相当于选择 \(n-1\) 个数存在 \(x_n\) 使得 \((\sum_{1\leq i < n } x_i ) + t x_n \equiv 0 \pmod m\) ,设 \(g = \gcd(t,m)\),则方案数 \(=f(n-1,g,1)\) ,对于 \(x_n\) ,我们也可以确定其取值个数,即 \(Mg/m\) 。

    (2) \(x_i(i<n)\) 中存至少一个与 \(x_n\) 相等的数。则在 \(x_1\cdots x_{n-1}\) 个中任选一个和 \(x_n\) 相等,有 \(n-1\) 种可能,乘上对应的方案数 \(f(n-1,m,t+1)\) 即可。

由于我们经常要用到 \(f(a,b,1)\) ,所以我们对于 \(f(a,b,1)\) 记忆化一下,然后搜索即可。注意 \(f(a,n,1)\) 这种要特殊处理。

代码

static const int N=1005,mod=1e9+7;
int M;
int dp[N][N],Fac[N];
int Pow(int x,int y){
int ans=1;
for (;y;y>>=1,x=1LL*x*x%mod)
if (y&1)
ans=1LL*ans*x%mod;
return ans;
}
int gcd(int x,int y){
return y?gcd(y,x%y):x;
}
int f(int n,int m,int t){
if (n==0)
return 1;
if (m==1)
return Fac[n];
int g=gcd(m,t);
// sum + t * x = 0
// c(sum = 0) - c(sum = 0 && (t+1))
if (t==1){
int _m=min(m,1000+1);
if (~dp[n][_m])
return dp[n][_m];
dp[n][_m]=((1LL*f(n-1,g,1)*(M/m*g)
-1LL*f(n-1,m,t+1)*(n-1))%mod+mod)%mod;
return dp[n][_m];
}
int res=((1LL*f(n-1,g,1)*(M/m*g)
-1LL*f(n-1,m,t+1)*(n-1))%mod+mod)%mod;
return res;
}
int find(int N, int K){
M=N;
memset(dp,-1,sizeof dp);
int t=1;
Fac[0]=1;
for (int i=1;i<=K;i++){
t=1LL*t*i%mod;
Fac[i]=1LL*Fac[i-1]*(N-i+1)%mod;
}
t=Pow(t,mod-2);
int ans=1LL*t*f(K,N,1)%mod;
return ans;
}

TopCoder SRM502 Div1 1000 动态规划的更多相关文章

  1. TopCoder SRM500 Div1 1000 其他

    原文链接https://www.cnblogs.com/zhouzhendong/p/SRM500-1000.html SRM500 Div1 1000 设 \(v_1,v_2,\cdots ,v_9 ...

  2. TopCoder SRM502 Div1 500 贪心 01背包

    原文链接https://www.cnblogs.com/zhouzhendong/p/SRM502-500.html SRM502 Div1 500 好题. 首先,如果已经确定了解决所有问题的优先级, ...

  3. TopCoder 649 div1 & div2

    最近一场TC,做得是在是烂,不过最后challenge阶段用一个随机数据cha了一个明显错误的代码,最后免于暴跌rating,还涨了一点.TC题目质量还是很高的,非常锻炼思维,拓展做题的视野,老老实实 ...

  4. TopCoder SRM500 Div1 250 其他

    原文链接https://www.cnblogs.com/zhouzhendong/p/SRM500-250.html SRM500 Div1 250 题意 (看题用了半个小时--) 有 n 个人(编号 ...

  5. TopCoder SRM500 Div1 500 分治

    原文链接https://www.cnblogs.com/zhouzhendong/p/SRM500-500.html SRM500 Div1 500 没想到 double 的精度居然没有爆-- 考虑以 ...

  6. topcoder SRM712 Div1 LR

    题目: Problem Statement      We have a cyclic array A of length n. For each valid i, element i-1 the l ...

  7. TopCoder 603 div1 & div2

    div2 250pts MiddleCode 题意:s串长度为奇数时,将中间字符取掉并添加到t末尾:长度为偶数时,将中间两个较小的字符取掉并添加到末尾. 分析:直接做,学习了一下substr(s, p ...

  8. TopCoder SRM704 Div1 800 构造

    原文链接https://www.cnblogs.com/zhouzhendong/p/SRM704-800.html 题解 考虑构造一个 $n = 20$ 的图. 先把所有 $i$ 都连向 $i-1$ ...

  9. topcoder SRM642 div1 hard WheelofFortune

    题目链接:vjudge 大意:有两个人参加一场游戏,这个游戏在一个编号为\(0\text~n-1\)的轮盘上进行,一开始轮盘上的数字均为0:一共有\(m\)轮,每一轮都有一个操作参数\(s_i\),主 ...

随机推荐

  1. Linux下怎样搜索文件

    使用linux系统难免会忘记文件所在的位置,可以使用以下命令对系统中的文件进行搜索.搜索文件的命令为”find“:”locate“:”whereis“:”which“:”type“   方法/步骤   ...

  2. boost.Asio lib

    Documentation for Boost.Asio http://www.boost.org/doc/libs/1_62_0/doc/html/boost_asio.html https://w ...

  3. 腾讯云CVM之间配置免密钥登录

    背景: 1客户A和B俩台主机之间需要实现免密钥登录,已绑定腾讯云申请的密钥对 系统:centos7.3 A:192.168.0.100 B:192.168.0.84 A主机的私钥文件:aaa B主机的 ...

  4. C#遍历指定文件夹中的所有文件(转)

    C#遍历指定文件夹中的所有文件 DirectoryInfo TheFolder=new DirectoryInfo(folderFullName);//遍历文件夹foreach(DirectoryIn ...

  5. 相关子查询和嵌套子查询 [SQL Server]

    SQLServer子查询可以分为 相关子查询 和 嵌套子查询 两类.前提,假设Books表如下: 类编号  图书名         出版社               价格-------------- ...

  6. idea 去除xml文件sql语句背景色

    怎么看idea mapper.xml中写sql语句的那种屎黄屎黄背景颜色不好看 去除背景色 把这两项勾选去掉,然后 把这个背景勾选也去掉,最后 把这个勾选也去掉 另一种方式: 也可以使用这种方式 al ...

  7. url加密和解密

    .NET中加密和解密有两种方式 string file="文件上(传)篇.doc";string Server_UrlEncode=Server.UrlEncode(file);s ...

  8. 银联支付Java开发

    注:原来来源于: <  银联支付Java开发  > 银联的demo写的不错,基本上可以直接使用. 首先是对acp_sdk.properties的内容修改,注意这个文件的文件名不能进行修改. ...

  9. js原生回到顶部,并平滑过渡---- 记录

    window.scrollTo({ top: 0, behavior: 'smooth', });

  10. linux学习笔记:第三单元 Linux命令及获取帮助

    第三单元 Linux命令及获取帮助 11) 了解Linux命令的语法格式:命令 [选项] [参数]2) 掌握命令格式中命令.选项.参数的具体含义a) 命令:告诉Linux(UNIX)操作系统做(执行) ...