洛谷 P4389 付公主的背包 解题报告
P4389 付公主的背包
题目背景
付公主有一个可爱的背包qwq
题目描述
这个背包最多可以装\(10^5\)大小的东西
付公主有\(n\)种商品,她要准备出摊了
每种商品体积为\(V_i\),都有\(10^5\)件
给定\(m\),对于\(s\in [1,m]\),请你回答用这些商品恰好装\(s\)体积的方案数
输入输出格式
输入格式:
第一行\(n,m\)
第二行\(V_1\sim V_n\)
输出格式:
\(m\)行,第\(i\)行代表\(s=i\)时方案数,对\(998244353\)取模
说明
对于\(30\%\)的数据,\(n\le 3000,m\le 3000\)
对于\(60\%\)的数据,纯随机生成
对于\(100\%\)的数据, \(n\le 100000,m\le 100000\)
对于\(100\%\)的数据,\(V_i\le m\)
先构造一波生成函数
\]
然后答案是\(n\)个东西卷起来,复杂度高达\(O(nm\log m)\),显然苟不住
不妨把\(n\)个函数都取对数,然后就成了多项式加法,可以直接枚举倍数做到\(O(m\ln m)\),现在考虑如何转换成为对数
G&=\ln F\\
G'&=\frac{F'}{F}\\
&=(1-x^v)\sum_{i=0}^\infty vix^{vi-1}\\
&=\sum_{i=0}^\infty vix^{vi-1}-\sum_{i=0}^\infty vix^{v(i+1)-1}\\
&=\sum_{i=0}^\infty vix^{vi-1}-\sum_{i=0}^\infty v(i-1)x^{vi-1}\\
&=\sum_{i=0}^\infty vix^{vi-1}\\
G&=\int G'\\
&=\sum_{i=0}^\infty \frac{1}{i}x^{vi}
\end{aligned}
\]
于是我们需要实现的就只有多项式exp啦
Code:
#include <cstdio>
#include <algorithm>
const int N=(1<<18)+10;
const int mod=998244353,Gi=332748118;
#define mul(a,b) (1ll*(a)*(b)%mod)
#define add(a,b) ((a+b)%mod)
int ans[N],G[N],turn[N],ina[N],inb[N],lna[N],lnb[N],exa[N],exb[N],cnt[N],Inv[N];
int qp(int d,int k){int f=1;while(k){if(k&1)f=mul(f,d);d=mul(d,d),k>>=1;}return f;}
void NTT(int *a,int typ,int len)
{
int L=-1;for(int i=1;i<len;i<<=1) ++L;
for(int i=0;i<len;i++)
{
turn[i]=turn[i>>1]>>1|(i&1)<<L;
if(i<turn[i]) std::swap(a[i],a[turn[i]]);
}
for(int le=1;le<len;le<<=1)
{
int wn=qp(typ?3:Gi,(mod-1)/(le<<1));
for(int p=0;p<len;p+=le<<1)
{
int w=1;
for(int i=p;i<p+le;i++,w=mul(w,wn))
{
int tx=a[i],ty=mul(w,a[i+le]);
a[i]=add(tx,ty);
a[i+le]=add(tx,mod-ty);
}
}
}
if(!typ)
{
int inv=qp(len,mod-2);
for(int i=0;i<len;i++) a[i]=mul(a[i],inv);
}
}
void polyinv(int *a,int *b,int len)
{
if(len==1){b[0]=qp(a[0],mod-2);return;}
polyinv(a,b,len>>1);
for(int i=0;i<len<<1;i++) ina[i]=inb[i]=0;
for(int i=0;i<len;i++) ina[i]=b[i],inb[i]=a[i];
NTT(ina,1,len<<1),NTT(inb,1,len<<1);
for(int i=0;i<len<<1;i++) ina[i]=mul(ina[i],add(2,mod-mul(ina[i],inb[i])));
NTT(ina,0,len<<1);
for(int i=0;i<len;i++) b[i]=ina[i];
}
void inter(int *a,int n){for(int i=n;i;i--)a[i]=mul(a[i-1],Inv[i]);a[0]=0;}
void drev(int *a,int n){for(int i=0;i<n;i++)a[i]=mul(a[i+1],i+1);a[n]=0;}
void polyln(int *a,int *b,int len)
{
for(int i=0;i<len<<1;i++) lna[i]=lnb[i]=0;
for(int i=0;i<len;i++) lna[i]=a[i];
polyinv(lna,lnb,len);
drev(lna,len-1);
NTT(lna,1,len<<1),NTT(lnb,1,len<<1);
for(int i=0;i<len<<1;i++) lna[i]=mul(lna[i],lnb[i]);
NTT(lna,0,len<<1);
inter(lna,len-1);
for(int i=0;i<len;i++) b[i]=lna[i];
}
void polyexp(int *a,int *b,int len)
{
if(len==1){b[0]=1;return;}
polyexp(a,b,len>>1);
for(int i=0;i<len<<1;i++) exa[i]=exb[i]=0;
polyln(b,exa,len);
for(int i=0;i<len;i++) exa[i]=add(a[i]+(i==0),mod-exa[i]);
for(int i=0;i<len;i++) exb[i]=b[i];
NTT(exa,1,len<<1),NTT(exb,1,len<<1);
for(int i=0;i<len<<1;i++) exa[i]=mul(exa[i],exb[i]);
NTT(exa,0,len<<1);
for(int i=0;i<len;i++) b[i]=exa[i];
}
int main()
{
int n,m;scanf("%d%d",&n,&m);
for(int v,i=1;i<=n;i++) scanf("%d",&v),++cnt[v];
++m;int len=1;while(len<m) len<<=1;
Inv[0]=1;for(int i=1;i<=len;i++) Inv[i]=qp(i,mod-2);
for(int i=1;i<=m;i++)
for(int j=i;j<=m;j+=i)
G[j]=add(G[j],mul(cnt[i],Inv[j/i]));
polyexp(G,ans,len);
for(int i=1;i<m;i++) printf("%d\n",ans[i]);
return 0;
}
2018.12.29
洛谷 P4389 付公主的背包 解题报告的更多相关文章
- 洛谷 P4389: 付公主的背包
题目传送门:洛谷 P4389. 题意简述: 有 \(n\) 个物品,每个物品都有无限多,第 \(i\) 个物品的体积为 \(v_i\)(\(v_i\le m\)). 问用这些物品恰好装满容量为 \(i ...
- 洛谷P4389 付公主的背包--生成函数+多项式
题目链接戳这里 题目描述 有\(n\)件不同的商品,每件物品都有无限个,输出总体积为\([1,m]\)的方案数 思路 直接跑背包有\(30\) 考虑把每个物品的生成函数设出来,对于一件体积为\(v\) ...
- 洛谷P4389 付公主的背包 [生成函数,NTT]
传送门 同样是回过头来发现不会做了,要加深一下记忆. 思路 只要听说过生成函数的人相信第一眼都可以想到生成函数. 所以我们要求 \[ ans=\prod \sum_n x^{nV}=\prod \fr ...
- [洛谷P4389]付公主的背包
题目大意:有$n(n\leqslant10^5)$种物品,第$i$个物品体积为$v_i$,都有$10^5$件.给定$m(m\leqslant10^5)$,对于$s\in [1,m]$,请你回答用这些商 ...
- 洛谷 4389 付公主的背包——多项式求ln、exp
题目:https://www.luogu.org/problemnew/show/P4389 关于泰勒展开: https://blog.csdn.net/SoHardToNamed/article/d ...
- 洛谷 P4495 [HAOI2018]奇怪的背包 解题报告
P4495 [HAOI2018]奇怪的背包 题目描述 小\(C\)非常擅长背包问题,他有一个奇怪的背包,这个背包有一个参数\(P\),当他 向这个背包内放入若干个物品后,背包的重量是物品总体积对\(P ...
- 洛谷 P2323 [HNOI2006]公路修建问题 解题报告
P2323 [HNOI2006]公路修建问题 题目描述 输入输出格式 输入格式: 在实际评测时,将只会有m-1行公路 输出格式: 思路: 二分答案 然后把每条能加的大边都加上,然后加小边 但在洛谷的题 ...
- luogu P4389 付公主的背包
传送门 神仙题鸭!orz dkw 暴力就是完全背包 而完全背包可以和生成函数扯上关系,记第i种物品质量为\(a_i\),那么这种物品的生成函数\(G(i)=\sum_{j=0}^{\infty}x^{ ...
- P4389 付公主的背包
注意 初始化的时候要这样写 for(int i=1,x;i<=n;i++){ scanf("%d",&x); v[x]++; } for(int i=1;i<= ...
随机推荐
- 20155330 《网络对抗》 Exp9 web安全基础实践
20155330 <网络对抗> Exp9 web安全基础实践 实验问题回答 SQL注入攻击原理,如何防御 原理:SQL注入攻击值得是通过构建特殊的输入作为参数传入web应用程序,而这些输入 ...
- POJ1035&&POJ3080&&POJ1936
字符串处理专题,很早就写好了然而忘记写blog了 1035 题意:给你一些单词作为字典.然后让你查找一些单词.对于每个单词,如果在字典中就输出它.否则输出所有它通过删除||增加||替换一个字符能得到的 ...
- MFC CTreeCtrl运用
CTreeCtrl运用 删除无效资源 递归的运用 自写遍历目录函数 递归遍历所有子目录 一.删除无效资源 .打开资源文件 .找到无效链接删掉 二.自写遍历目录函数 CFileFind findfile ...
- Google是如何教会机器玩Atari游戏的
转自:http://blog.csdn.net/revolver/article/details/50177219 今年上半年(2015年2月),Google在Nature上发表了一篇论文:Human ...
- [CF1038F]Wrap Around[AC自动机+dp]
题意 题目链接 分析 题意容易转化成求循环之后不包含 \(s\) 的串的个数. 首先建立 AC 自动机.考虑一个暴力的做法:枚举长度为 \(n\) 的字符串 \(t\) 最终(后缀) 和 \(s\) ...
- 本地navicat远程连接到云服务器数据库
本来没有开启秘钥的远程服务器端数据库连接非常方便,就在新建连接上填入数据就ok了,但是开启SSH秘钥后的服务器连接有一个大坑,下面来详细讲讲. 其实开启了秘钥,在新建连接下,先选择SSH方式登录到远程 ...
- 部署AlwaysOn第二步:配置AlwaysOn,创建可用性组
AlwaysOn是在SQL Server 2012中新引入的一种高可用技术,从名称中可以看出,AlwaysOn的设计目标是保持数据库系统永远可用.AlwaysOn利用了Windows服务器故障转移集群 ...
- TDD 与 CI 在 Python 中的实践
社区化产品的长久生存之道可能莫过于对迭代周期的控制.还记得以前采用老土的阶段开发的年代,将软件生命周期分为各个阶段,当到达每个阶段的里程碑则集中所有的资源.人力作全面冲刺.每次到了里程碑的检查点冲过了 ...
- Unity 2D相机公式换算(从其他博客上抄的)
2d camera, unit坐标,单位换算 2d游戏可以使用平行投影的camera,这种camera需要设置size (orthographicSize),size的含义为屏幕高度的一半,不过单位不 ...
- 1065 A+B and C
同样是一题会产生溢出的题,不同的是这个用大数类很麻烦,因为有负数的可能性 不妨利用溢出的特性来解题:两个整数和为负数 或者 两个负数和为正数,那么就溢出了. #include<bits/stdc ...