题意:求一个较大的多重背包对于每个i的方案数,答案对998244353取模。

思路:

生成函数:

对于一个\(V\)

设:

\(f(x) = \sum_{i=0}^{oo} x ^ {V * i} = {1 \over {1 - x ^ V}}\)

那么就是求这个生成函数的积。

首先将\(f(x)\)取\(ln\)为\(g(x)\),最后\(exp\)回去得到答案。

\(g'(x) = {f'(x) \over f(x)} = (1 - x^V)\sum_{i = 1}^{oo}V * i * x ^ {V}\)

最终\(exp\)一遍得到答案。

时间复杂度\(O(mlogm)\)

#include <bits/stdc++.h>
using namespace std;
const int maxn = 5000010;
const int mod = 998244353;
const int g = 3;
#define ll long long
int A[maxn];
int B[maxn];
int c[maxn],d[maxn];
int e[maxn];
int f[maxn];
int w[maxn][2];
int rev[maxn];
int cnt[maxn];
int v[maxn];
int ans[maxn];
int inv[maxn];
const int vg = (mod + 1) / 3;
int n,m,l;
int len;
inline int read() {
int q=0,f=1;char ch = getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;ch=getchar();
}
while(isdigit(ch)){
q=q*10+ch-'0';ch=getchar();
}
return q*f;
}
inline int pow_mod (int x,int y) {
int res = 1;
while(y) {
if(y & 1) res = (ll) res * x % mod;
x = (ll) x * x % mod;
y >>= 1;
}
return res;
}
inline void NTT(int *a,int n,int type) {
for(len = 1,l = 0;len <= n;len <<= 1, ++l);
for(int i = 0;i < len; ++i) {
rev[i] = (rev[i >> 1] >> 1) | ((i & 1) << (l - 1));
}
for(int i = 0;i < len; ++i) {
if(i < rev[i]) {
swap(a[i],a[rev[i]]);
}
}
for(int i = 1;i < len; i <<= 1) {
int wn = (~type) ? w[i][0]:w[i][1];
for(int j = 0;j < len;j += (i << 1)) {
int w = 1;
for(int k = 0;k < i; ++k,w = (ll)w * wn % mod) {
int x = (ll)a[i + j + k];
int y = a[j + k];
a[j + k] = (x + y) % mod;
a[i + j + k] = (y + mod - x) % mod;
}
}
}
if(type == -1) {
int tmp = pow_mod(len,mod - 2);
for(int i = 0;i < len; ++i) {
a[i] = (ll)a[i] * tmp % mod;
}
}
}
inline void get_Inv(int *a,int *b,int n) {
if(n == 1) {
b[0] = pow_mod(a[0],mod - 2);
return;
}
get_Inv(a,b,n >> 1);
for(int i = 0;i < n; ++i) {
A[i] = a[i];
B[i] = b[i];
}
NTT(A,n << 1,1);
NTT(B,n << 1,1);
for(int i = 0;i < len; ++i) {
A[i] = (ll)A[i] * B[i] % mod * B[i] % mod;
}
NTT(A,n << 1,-1);
for(int i = 0;i < n; ++i) {
b[i] = ((b[i] << 1) % mod - A[i] + mod) % mod;
}
for(int i = 0;i < len; ++i) {
A[i] = B[i] = 0;
}
} inline void get_ln(int *a,int *b,int n) {
get_Inv(a,c,n);
for(int i = 0;i < n - 1; ++i) {
d[i] = (ll)(i + 1) * a[i + 1] % mod;
}
NTT(c,n << 1,1);
NTT(d,n << 1,1);
for(int i = 0;i < len; ++i) {
c[i] = (ll) c[i] * d[i] % mod;
}
NTT(c,n << 1,-1);
for(int i = 1;i < n; ++i) {
b[i] = (ll)inv[i] * c[i - 1] % mod;
}
for(int i = 0;i < len; ++i) {
c[i] = d[i] = 0;
}
} inline void get_exp(int *a,int *b,int n) {
if(n == 1) {
b[0] = 1;
return;
}
get_exp(a,b,n >> 1);
for(int i = 0;i < n; ++i) {
e[i] = b[i];
}
get_ln(b,f,n);
for(int i = 0;i < n; ++i) {
f[i] = (mod - f[i] + a[i]) % mod;
f[0] = (f[0] + 1)%mod;
}
NTT(e,n << 1,1);
NTT(f,n << 1,1);
for(int i = 0;i < len; ++i) {
e[i] = (ll)e[i] * f[i] % mod;
}
NTT(e,n << 1,-1);
for(int i = 0;i < n; ++i) {
b[i] = e[i];
}
for(int i = 0;i < len; ++i) {
e[i] = f[i] = 0;
}
} int main () {
n = read(),m = read();
inv[1] = 1;
for(int i = 1;i <= n; ++i) {
++cnt[read()];
}
int tmp = 1;
for(;tmp <= m;tmp <<= 1);
for(int i = 1;i <= (tmp << 1);i <<= 1) {
w[i][0] = pow_mod(g,(mod - 1)/(i << 1));
w[i][1] = pow_mod(vg,(mod - 1)/(i<<1));
}
for(int i = 2;i <= tmp; ++i) {
inv[i] = (ll)(mod - mod / i) * inv[mod % i]%mod;
}
for(int i = 1;i <= m; ++i) {
if(cnt[i]) {
int dl = (ll) i * cnt[i] % mod;
for(int j = 1;i * j <= m; ++j) {
v[i * j] = (v[i * j] + dl) % mod;
}
}
}
for(int i = 1;i <= m; ++i) {
v[i] = (ll)inv[i] * v[i] % mod;
}
get_exp(v,ans,tmp);
for(int i = 1;i <= m; ++i) {
printf("%d\n",ans[i] % mod);
}
return 0;
}
//原地爆炸

[luogu 4389] 付公主的背包的更多相关文章

  1. luogu P4389 付公主的背包

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

  2. 洛谷 4389 付公主的背包——多项式求ln、exp

    题目:https://www.luogu.org/problemnew/show/P4389 关于泰勒展开: https://blog.csdn.net/SoHardToNamed/article/d ...

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

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

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

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

  5. luoguP4389 付公主的背包

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

  6. [luogu4389]付公主的背包(多项式exp)

    完全背包方案计数问题的FFT优化.首先写成生成函数的形式:对重量为V的背包,它的生成函数为$\sum\limits_{i=0}^{+\infty}x^{Vi}=\frac{1}{1-x^{V}}$于是 ...

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

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

  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. 分布式锁实现(一):Redis

    前言 单机环境下我们可以通过JAVA的Synchronized和Lock来实现进程内部的锁,但是随着分布式应用和集群环境的出现,系统资源的竞争从单进程多线程的竞争变成了多进程的竞争,这时候就需要分布式 ...

  2. 【JZOJ3673】【luoguP4040】【BZOJ3874】宅男计划

    description 外卖店一共有N种食物,分别有1到N编号.第i种食物有固定的价钱Pi和保质期Si.第i种食物会在Si天后过期.JYY是不会吃过期食物的. 比如JYY如果今天点了一份保质期为1天的 ...

  3. go变量和声明

    go变量和声明 这将是美好的开始和结束,通过写下x = 4,我们查看变量,可以说声明了一个变量并赋值,但是很不幸,go语言变量声明和赋值比这更复杂.通过学习一些简单的示例开始学习变量声明和赋值.然后在 ...

  4. SQLServer AlwaysOn在阿里云的前世今生

    缘起 早在2015年的时候,随着阿里云业务突飞猛进的发展,SQLServer业务也积累了大批忠实客户,其中一些体量较大的客户在类似大促的业务高峰时RDS的单机规格(规格是按照 内存CPUIOPS 一定 ...

  5. delphi 文件夹操作(监控)

    delphi 监控文件系统 delphi 监控文件系统 你是否想为你的Windows加上一双眼睛,察看使用者在机器上所做的各种操作(例如建立.删除文件:改变文件或目录名字)呢? 这里介绍一种利用Win ...

  6. Codeforces 1173B Nauuo and Chess

    题目链接:http://codeforces.com/problemset/problem/1173/B 思路参考:https://www.cnblogs.com/blowhail/p/1099123 ...

  7. 【Java多线程系列一】Java实现线程方法

    Java实现线程的两种方法 继承Thread类 实现Runnable接口 它们之间的区别如下: Java的类为单继承,但可以实现多个接口,因此Runnable可能在某些场景比Thread更适用 Thr ...

  8. JSoup安装

    要运行任何jsoup示例,需要先安装好jsoup相关Jar包.到目前为止(2017年05月),jsoup的当前版本是1.10.2.0.安装jsoup主要有三种方法: 通过Maven的pom.xml配置 ...

  9. mysql shell脚本

    mysql shell连接脚本 本地连接及远程链接 #!/bin/bash #连接MySQL数据库 Host=127.0.0.1 User=username PASSWORD=password POR ...

  10. js图片放大境效果

    放大境效果如下图所示,当鼠标放到小图时,就会出现浅黄色的小块,而右边方框也出现了,并且右边方框的内容时根据浅黄色小块的内容变换而变换: 原理: 1,准备2张图,一大一小,如上图所示,小图的盒子div1 ...