这个题太神辣~


暴力背包就能获得\(30\)分的好成绩......

\(60\)分不知道咋搞.....

所以直接看\(100\)分吧\(QwQ\)

用一点生成函数的套路,对于一个体积为\(v\)的物品,我们构造一个序列\(f_n = [v \mid n]\ (n \ge 0)\)

其生成函数\(F(x) = \sum\limits_{i=0}^{\infty} [v \mid i]x^i = \sum\limits_{i=0}^{\infty} x^{vi} = \frac{1}{1-x^v}\)

那么答案就是这\(n\)个\(F\)的乘积。

考虑怎么把这个东西乘起来,大力乘是\(O(nm \log m)\),其封闭形式\(\frac{1}{1-x^v}\)好像也没有什么好优化的......

一个套路的想法是把每个\(F\),先\(\ln\)一下,然后再加起来,最后\(\exp\)回去。

问题变成了怎么求\(\ln F(x)\),不放设\(\ln F(x) = G(x)\)。

套路的两边对\(x\)求导

\[\frac{F'(x)}{F(x)} = G'(x)
\]

注意到\(F(x) = \frac{1}{1-x^v}\)

\[\begin{aligned}
(1 - x^v)\sum\limits_{i=0}^{\infty} vix^{vi-1} &= G'(x) \\
\sum\limits_{i=0}^{\infty} vix^{vi-1} - \sum\limits_{i=0}^{\infty} vix^{v(i+1)-1} &= G'(x)
\end{aligned}
\]

注意到\(i=0\)时候两个\(\Sigma\)都是\(0\),所以

\[\begin{aligned}
\sum\limits_{i=1}^{\infty} vix^{vi-1} - \sum\limits_{i=1}^{\infty} v(i-1) x^{vi-1} &= G'(x) \\
\sum\limits_{i=1}^{\infty} vx^{vi-1} &= G'(x)
\end{aligned}
\]

两边同时积分回去

\[G(x) = \int\sum\limits_{i=1}^{\infty} vx^{vi-1}dx = \sum\limits_{i=1}^{\infty}v \cdot \frac{1}{vi}x^{vi} = \sum\limits_{i=1}^{\infty} \frac{1}{i}x^{vi}
\]

可以对每个\(v\),\(O(\frac{m}{v})\)的搞出\(G\),然后\(\exp\)回去就好了。

\(v\)有重复的话开个桶可以保证复杂度,预处理逆元能快一点。

\(Code:\)

#include <bits/stdc++.h>
using namespace std;
const int N=3e5+10,P=998244353,gen=3,igen=(P+1)/gen;
int add(int x,int y){return x+y>=P?x+y-P:x+y;}
int sub(int x,int y){return x-y<0?x-y+P:x-y;}
int fpow(int x,int y){
int ret=1; for(x%=P;y;y>>=1,x=1ll*x*x%P)
if(y&1) ret=1ll*x*ret%P;
return ret;
}
namespace Poly{
int rev[N];
void init(int n){
for(int i=0;i<n;i++)
rev[i]=rev[i>>1]>>1|((i&1)?n>>1:0);
}
void ntt(int *f,int n,int flg){
for(int i=0;i<n;i++)
if(rev[i]<i) swap(f[i],f[rev[i]]);
for(int len=2,k=1;len<=n;len<<=1,k<<=1){
int wn=fpow(flg==1?gen:igen,(P-1)/len);
for(int i=0;i<n;i+=len)
for(int j=i,w=1;j<i+k;j++,w=1ll*w*wn%P){
int tmp=1ll*w*f[j+k]%P;
f[j+k]=sub(f[j],tmp),f[j]=add(f[j],tmp);
}
}
if(flg==-1){
int inv=fpow(n,P-2);
for(int i=0;i<n;i++) f[i]=1ll*f[i]*inv%P;
}
}
#define glim(n) int limit=1;while(limit<=(n))limit<<=1;init(limit)
void getinv(int *f,int n,int *G){
if(n==1){G[0]=fpow(f[0],P-2);return;}
getinv(f,(n+1)>>1,G); glim(2*n);
static int F[N];
for(int i=0;i<limit;i++)
i>=n?F[i]=G[i]=0:F[i]=f[i],G[i]=G[i];
ntt(F,limit,1),ntt(G,limit,1);
for(int i=0;i<limit;i++) G[i]=1ll*G[i]*sub(2,1ll*F[i]*G[i]%P)%P;
ntt(G,limit,-1);
for(int i=n;i<limit;i++) G[i]=0;
}
void dao(int *f,int n,int *G){
static int F[N]; for(int i=0;i<=n;i++) F[i]=f[i];
for(int i=1;i<=n;i++) G[i-1]=1ll*F[i]*i%P; G[n]=0;
}
void jifen(int *f,int n,int *G){
static int F[N]; for(int i=0;i<=n;i++) F[i]=f[i];
for(int i=0;i<=n;i++) G[i+1]=1ll*F[i]*fpow(i+1,P-2)%P; G[0]=0;
}
void getln(int *f,int n,int *G){
static int F[N],iF[N]; for(int i=0;i<n;i++) F[i]=f[i];
getinv(F,n,iF),dao(F,n-1,F); glim(n*2);
for(int i=0;i<limit;i++) F[i]=i>=n?0:F[i],iF[i]=i>=n?0:iF[i];
ntt(F,limit,1),ntt(iF,limit,1);
for(int i=0;i<limit;i++) G[i]=1ll*F[i]*iF[i]%P;
ntt(G,limit,-1),jifen(G,n-1,G);
for(int i=n;i<limit;i++) G[i]=0;
}
void getexp(int *f,int n,int *G){
if(n==1){G[0]=1;return;}
getexp(f,(n+1)>>1,G); glim(n*2);
static int F[N],lnG[N];
for(int i=0;i<limit;i++) i>=n?F[i]=G[i]=0:F[i]=f[i],G[i]=G[i];
getln(G,n,lnG),ntt(F,limit,1),ntt(G,limit,1),ntt(lnG,limit,1);
for(int i=0;i<limit;i++) G[i]=1ll*G[i]*add(sub(1,lnG[i]),F[i])%P;
ntt(G,limit,-1);
for(int i=n;i<limit;i++) G[i]=0;
}
}
int F[N],ans[N],inv[N],cnt[N];
int main(){
int n,m; scanf("%d%d",&n,&m);
inv[1]=1; for(int i=2;i<=m;i++)
inv[i]=1ll*(P-P/i)*inv[P%i]%P;
for(int _=1;_<=n;_++){
int v;scanf("%d",&v);
if(v<=m) cnt[v]=add(cnt[v],1);
}
for(int v=1;v<=m;v++) if(cnt[v])
for(int i=1;i*v<=m;i++) F[i*v]=add(F[i*v],1ll*cnt[v]*inv[i]%P);
Poly::getexp(F,m+1,ans);
for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
return 0;
}

[题解] LuoguP4389 付公主的背包的更多相关文章

  1. LuoguP4389 付公主的背包【生成函数+多项式exp】

    题目背景 付公主有一个可爱的背包qwq 题目描述 这个背包最多可以装10^5105大小的东西 付公主有n种商品,她要准备出摊了 每种商品体积为Vi,都有10^5105件 给定m,对于s\in [1,m ...

  2. luoguP4389 付公主的背包

    luogu 显然这是个背包题 显然物品的数量是不用管的 所以考虑大小为\(v\)的物品可以装的体积用生成函数表示一下 \[ f(x)=\sum_{i=0}^{+\infty}x^{vi}=\frac{ ...

  3. luoguP4389 付公主的背包 多项式exp

    %%%dkw 话说这是个论文题来着... 考虑生成函数\(OGF\) 对于价值为\(v\)的物品,由于有\(10^5\)的件数,可以看做无限个 那么,其生成函数为\(x^0 + x^{v} + x^{ ...

  4. 洛谷 P4389 付公主的背包 解题报告

    P4389 付公主的背包 题目背景 付公主有一个可爱的背包qwq 题目描述 这个背包最多可以装\(10^5\)大小的东西 付公主有\(n\)种商品,她要准备出摊了 每种商品体积为\(V_i\),都有\ ...

  5. 洛谷 P4389: 付公主的背包

    题目传送门:洛谷 P4389. 题意简述: 有 \(n\) 个物品,每个物品都有无限多,第 \(i\) 个物品的体积为 \(v_i\)(\(v_i\le m\)). 问用这些物品恰好装满容量为 \(i ...

  6. 洛谷P4389 付公主的背包--生成函数+多项式

    题目链接戳这里 题目描述 有\(n\)件不同的商品,每件物品都有无限个,输出总体积为\([1,m]\)的方案数 思路 直接跑背包有\(30\) 考虑把每个物品的生成函数设出来,对于一件体积为\(v\) ...

  7. luogu P4389 付公主的背包

    传送门 神仙题鸭!orz dkw 暴力就是完全背包 而完全背包可以和生成函数扯上关系,记第i种物品质量为\(a_i\),那么这种物品的生成函数\(G(i)=\sum_{j=0}^{\infty}x^{ ...

  8. luogu4389 付公主的背包

    题目链接:洛谷 题目大意:现在有$n$个物品,每种物品体积为$v_i$,对任意$s\in [1,m]$,求背包恰好装$s$体积的方案数(完全背包问题). 数据范围:$n,m\leq 10^5$ 这道题 ...

  9. P3489 付公主的背包

    题意:n<=1e5,m<=1e5,跑n个物品1到m容量的完全背包. 考虑暴力的做法就是把一些1/(1+x^a)的多项式乘起来即可. 考虑优化,取一下ln,转化为加法,然后exp回去就好了.

随机推荐

  1. 关于Tomcat部署项目的点点滴滴

    在给客户部署环境时,我们不可能想开发一样,在编辑软件部署一下tomcat就可以正常运行.我们也应该清楚java的运行机制**“先编译,后解释”**的原则.(如下图)![图片描述][1]那么在Tomca ...

  2. Python基础week2

    本节内容 列表.元组操作 字符串操作 字典操作 集合操作 文件操作 字符编码与转码 1. 列表.元组操作 列表是我们最以后最常用的数据类型之一,通过列表可以对数据实现最方便的存储.修改等操作 定义列表 ...

  3. vscode点击ctrl键报错Request textDocument/definition failed.

    现象 用vscode写java代码的时候突然出现,修复问题点击Ctrl时,输出窗口就打日志,报错Request textDocument/definition failed. 我百度唯一的有用线索就是 ...

  4. C语言中指针和取地址符&的关系

    一 概念定义: 严格说起来,应该这么讲:指针存的是地址,而&运算符是取变量的地址. 指针原理: 其实计算机中的内存就像一个个抽屉(一兆内存就相当于1百万个抽屉),每个抽屉都有一个编号,用于存放 ...

  5. 使用WinDbg分析蓝屏dump原因

    大多数人或许都经历过系统蓝屏问题,然而大多数人不清楚该怎么处理蓝屏问题,这里主要对系统蓝屏做一些解释,同时介绍下蓝屏问题分析工具WinDbg分析蓝屏问题的一般步骤. 微软官方对蓝屏的定义是,当系统遇到 ...

  6. 十六、React 渲染数据注意事项、以及react-router4.x中使用js跳转路由(登录成功自动跳转首页)

    一.React加载数据流程回顾 先看上一节的产品详情代码:https://blog.csdn.net/u010132177/article/details/103184176 [Pcontent.js ...

  7. 吴裕雄 Bootstrap 前端框架开发——Bootstrap 字体图标(Glyphicons):glyphicon glyphicon-heart

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...

  8. UVA - 548 Tree(二叉树的递归遍历)

    题意:已知中序后序序列,求一个叶子到根路径上权和最小,如果多解,则叶子权值尽量小. 分析:已知中序后序建树,再dfs求从根到各叶子的权和比较大小 #include<cstdio> #inc ...

  9. 微信小程序实现左滑删除效果(原生/uni-app)

    实现效果 列表中侧滑删除 删除不同时存在 scrollview上下滑动与侧滑删除不影响 uni-app实现 html部分 <template> <scroll-view :scrol ...

  10. HihoCoder#1052:基因工程

    HihoCoder#1052:基因工程 时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho正在进行一项基因工程实验.他们要修改一段长度为N的DNA序列,使得这段 ...