2655: calc

Time Limit: 30 Sec  Memory Limit: 512 MB
Submit: 428  Solved: 246
[Submit][Status][Discuss]

Description

  一个序列a1,...,an是合法的,当且仅当:
  长度为给定的n。
  a1,...,an都是[1,A]中的整数。
  a1,...,an互不相等。
  一个序列的值定义为它里面所有数的乘积,即a1a2...an。
  求所有不同合法序列的值的和。
  两个序列不同当且仅当他们任意一位不一样。
  输出答案对一个数mod取余的结果。

Input

  一行3个数,A,n,mod。意义为上面所说的。

Output

  一行结果。

Sample Input

9 7 10007

Sample Output

3611

HINT

数据规模和约定

  0:A<=10,n<=10。

  1..3:A<=1000,n<=20.

  4..9:A<=10^9,n<=20

  10..19:A<=10^9,n<=500。

  全部:mod<=10^9,并且mod为素数,mod>A>n+1

Source

[Submit][Status][Discuss]

https://blog.csdn.net/qq_20669971/article/details/52790835

先列出DP方程,f[i][j]表示i个元素选j个进排列的总贡献,则f[i][j]=f[i-1][j-1]*i*j+f[i-1][j]。(这里也可以把n!提出来)

直接DP肯定不行,然后我们可以发现f[i][j]实际上是关于i的高次多项式,现在要确定是几次多项式。

第一种方法:f[i][i]=(i!)^2,根据递推式求得f[i][j]是2j次的。

https://www.cnblogs.com/xiao-ju-ruo-xjr/p/8510924.html

第二种方法:打表发现系数中i的次数和i本身的次数都为j。

https://blog.csdn.net/ez_yww/article/details/77221338

所以确定是2j次的(当然如果不想确定就多放几次也没关系)

这样我们可以用拉格朗日插值法先求出2n个点的f[x_i][n],拟合出多项式后将A代入即可(直接现场代入就好)

 #include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=l; i<=r; i++)
typedef long long ll;
using namespace std; const int N=;
int f[N<<][N],x,n,mod,ans,m; int inv(int a){
int res=,b=mod-;
for (; b; a=1ll*a*a%mod,b>>=)
if (b & ) res=1ll*res*a%mod;
return res;
} int main(){
freopen("bzoj2655.in","r",stdin);
freopen("bzoj2655.out","w",stdout);
scanf("%d%d%d",&x,&n,&mod); f[][]=; m=*n+;
rep(i,,m) { f[i][]=; rep(j,,n) f[i][j]=(f[i-][j]+1ll*i*f[i-][j-])%mod; }
rep(i,,m){
int a=f[i][n],b=;
rep(j,,m) if (i!=j) a=1ll*a*(x-j)%mod,b=1ll*b*(i-j)%mod;
a=1ll*a*inv(b)%mod; ans=(ans+a)%mod;
}
rep(i,,n) ans=1ll*ans*i%mod;
printf("%d\n",(ans+mod)%mod);
return ;
}

可以O(m)插值,预处理一些东西就好(不过瓶颈在于DP所以无所谓)

边界考虑清楚。

 #include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=l; i<=r; i++)
typedef long long ll;
using namespace std; const int N=;
int x,n,mod,ans,m,f[N<<][N],pre[N<<],suf[N<<],fac[N<<],fin[N<<]; int inv(int a){
int res=,b=mod-;
for (; b; a=1ll*a*a%mod,b>>=)
if (b & ) res=1ll*res*a%mod;
return res;
} int main(){
freopen("bzoj2655.in","r",stdin);
freopen("bzoj2655.out","w",stdout);
scanf("%d%d%d",&x,&n,&mod); m=*n+;
rep(i,,m) f[i][]=;
rep(i,,m) rep(j,,n) f[i][j]=(f[i-][j]+1ll*i*f[i-][j-])%mod;
pre[]=; rep(i,,m) pre[i]=1ll*pre[i-]*(x-i)%mod;//(x-1)*(x-2)*...*(x-i)
suf[]=(x-m)%mod; rep(i,,m) suf[i]=1ll*suf[i-]*(x-m+i)%mod;//(x-m)*(x-m+1)*...(x-m+i)
fac[]=; rep(i,,m) fac[i]=1ll*fac[i-]*i%mod;//1*2*...*i
fin[m]=inv(fac[m]); for (int i=m-; ~i; i--) fin[i]=1ll*fin[i+]*(i+)%mod;//1/(1*2*...*i)
rep(i,,m){
int a=1ll*f[i][n]*pre[i-]%mod*((i==m)?:suf[m-i-])%mod;
int b=1ll*fin[i-]%mod*fin[m-i]*(((m-i)&)?-:)%mod;
ans=(ans+1ll*a*b)%mod;
}
rep(i,,n) ans=1ll*ans*i%mod;
printf("%d\n",(ans+mod)%mod);
return ;
}

以及另一道用拉格朗日插值法简化的题目:[BZOJ4559][JLOI2016]成绩比较

http://www.cnblogs.com/nbwzyzngyl/p/8394921.html

[BZOJ2655]calc(拉格朗日插值法+DP)的更多相关文章

  1. bzoj千题计划269:bzoj2655: calc (拉格朗日插值)

    http://www.lydsy.com/JudgeOnline/problem.php?id=2655 f[i][j] 表示[1,i]里选严格递增的j个数,序列值之和 那么ans=f[A][n] * ...

  2. P4463 [集训队互测2012] calc 拉格朗日插值 dp 多项式分析

    LINK:calc 容易得到一个nk的dp做法 同时发现走不通了 此时可以考虑暴力生成函数. 不过化简那套不太熟 且最后需要求多项式幂级数及多项式exp等难写的东西. 这里考虑观察优化dp的做法. 不 ...

  3. BZOJ2655 Calc - dp 拉格朗日插值法

    BZOJ2655 Calc 参考 题意: 给定n,m,mod,问在对mod取模的背景下,从[1,m]中选出n个数相乘可以得到的总和为多少. 思路: 首先可以发现dp方程 ,假定dp[m][n]表示从[ ...

  4. [国家集训队] calc(动规+拉格朗日插值法)

    题目 P4463 [国家集训队] calc 集训队的题目真是做不动呀\(\%>\_<\%\) 朴素方程 设\(f_{i,j}\)为前\(i\)个数值域\([1,j]\),且序列递增的总贡献 ...

  5. bzoj4559[JLoi2016]成绩比较 容斥+拉格朗日插值法

    4559: [JLoi2016]成绩比较 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 261  Solved: 165[Submit][Status ...

  6. Matlab数值计算示例: 牛顿插值法、LU分解法、拉格朗日插值法、牛顿插值法

    本文源于一次课题作业,部分自己写的,部分借用了网上的demo 牛顿迭代法(1) x=1:0.01:2; y=x.^3-x.^2+sin(x)-1; plot(x,y,'linewidth',2);gr ...

  7. 拉格朗日插值法——用Python进行数值计算

    插值法的伟大作用我就不说了.... 那么贴代码? 首先说一下下面几点: 1. 已有的数据样本被称之为 "插值节点" 2. 对于特定插值节点,它所对应的插值函数是必定存在且唯一的(关 ...

  8. CPP&MATLAB实现拉格朗日插值法

    开始学习MATLAB(R和Python先放一放...),老师推荐一本书,看完基础就是各种算法...首先是各种插值.先说拉格朗日插值法,这原理楼主完全不懂的,查的维基百科,好久才看懂.那里讲的很详细,这 ...

  9. codeforces 622F. The Sum of the k-th Powers 拉格朗日插值法

    题目链接 求sigma(i : 1 to n)i^k. 为了做这个题这两天真是补了不少数论, 之前连乘法逆元都不知道... 关于拉格朗日插值法, 我是看的这里http://www.guokr.com/ ...

随机推荐

  1. 【BZOJ 1492】 [NOI2007]货币兑换Cash 斜率优化DP

    先说一下斜率优化:这是一种经典的dp优化,是OI中利用数形结合的思想解决问题的典范,通常用于优化dp,有时候其他的一些决策优化也会用到,看待他的角度一般有两种,但均将决策看为二维坐标系上的点,并转化为 ...

  2. 【BZOJ 1146】[CTSC2008]网络管理Network

    树剖+树状数组套线段树O(nlogn^3)(我打的),有一种更加优秀的算法是O(nlogn^2)的就是直接树状数组套线段树欧拉序(并不快),或者是用主席树维护原始的树的信息,同时用树状数组套线段树维护 ...

  3. 【BZOJ 1070】[SCOI2007]修车 费用流

    就是拆个点限制一下(两点一排一大片),这道题让我注意到了限制这个重要的词.我们跑网络流跑出来的图都是有一定意义的,一般这个意义就对应了问题的一种方案,一般情况下跑一个不知道对不对的方案是相对容易的我们 ...

  4. React & Redux 的一些基本知识点

    一.React.createClass 跟 React.Component 的区别在于后者使用了ES6的语法,用constructor构造器来构造默认的属性和状态. 1. React.createCl ...

  5. 如何设置项目encoding为utf-8

    1.鼠标右键点击项目,选择[properties] 2.选择[Resource],在Text file encoding里面选择UTF-8,点击[ok] 大功告成! 木头大哥所发的文章均基于自身实践, ...

  6. oracle数据库导入导出方法

    Oracle Database 10g以后引入了最新的数据泵(Data Dump)技术,使DBA或开发人员可以将数据库元数据(对象定义)和数据快速移动到另一个oracle数据库中. 数据泵导出导入(E ...

  7. 正确答案 [Hash/枚举]

    正确答案 题目描述 小H与小Y刚刚参加完UOIP外卡组的初赛,就迫不及待的跑出考场对答案. "吔,我的答案和你都不一样!",小Y说道,"我们去找神犇们问答案吧" ...

  8. 湖南大学第十四届ACM程序设计新生杯 Dandan's lunch

    Dandan's lunch Description: As everyone knows, there are now n people participating in the competiti ...

  9. codeforces803D. Magazine Ad

    D. Magazine Adtime limit per test1 secondmemory limit per test256 megabytesinputstandard inputoutput ...

  10. -webkit-overflow-scrolling:touch;

    -webkit-overflow-scrolling建了带有硬件加速的系统级控件,所以效率很高.但是这相对是耗更多内存的,最好在产生了非常大面积的overflow时才应用. 而且在 ios8  里有b ...