Live2D

Solution -「洛谷 P4389」付公主的背包

Description

  Link.

  容量为 nm 种物品的无限背包,求凑出每种容量的方案数,对 998244353 取模。

  n,m105

Solution

  感觉货币系统是这道题的弱化版 qwq。

  还有这个博客园对齐公式自动编号的 feature 怎么去掉啊……

  对于大小为 v 的物品,有生成函数:

G(x)=i=0+xiv

  设物品 i 的生成函数为 Gi(x),则需要求:

i=1nGi(x)

  这个算不动,推一下式子:

(1)i=1nGi(x)=explni=1nGi(x)(2)=expi=1nlnGi(x)

  lnG(x) 拿出来:

(3)lnG(x)=lni=0+xiv(4)=ln11xv(5)=ln(1xv)(6)=d(1xv)1xv(7)=vxv11xvdx

  拿出积分里面的式子:

(8)vxv11xv=vxv1i=0+xiv(9)=vi=0+xv(i+1)1

  积分最后这项:

(10)vxv11xvdx=vi=0+xv(i+1)1dx(11)=vi=0+xv(i+1)v(i+1)(12)=i=1+xivi

  于是要求:

expi=1nj=1+xjvijmodxm+1

  先把 exp 内的式子写出来——枚举 v 的值域,计算 vi=v 时的贡献,最后 exp 一下就好啦。

  复杂度 O(n+mlogm)

Code

#include <cmath>
#include <cstdio>

typedef long long LL;

const int MAXN = 1 << 18, MOD = 998244353;
int n, m, inv[MAXN + 5], buc[MAXN + 5], F[MAXN + 5], G[MAXN + 5];

inline int add ( int a, const int b ) { return ( a += b ) < MOD ? a : a - MOD; }
inline int sub ( int a, const int b ) { return ( a -= b ) < 0 ? a + MOD : a; }
inline int mul ( LL a, const int b ) { return ( a *= b ) < MOD ? a : a % MOD; }

inline int rint () {
	int x = 0; char s = getchar ();
	for ( ; s < '0' || '9' < s; s = getchar () );
	for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
	return x;
}

inline int qkpow ( int a, int b ) {
	int ret = 1;
	for ( ; b; a = mul ( a, a ), b >>= 1 ) ret = mul ( ret, b & 1 ? a : 1 );
	return ret;
}

namespace Poly {

const int G = 3;

inline int adjust ( const int n ) {
	int ret = 0;
	for ( int l = 1; l < n; l <<= 1, ++ ret );
	return ret;
}

inline void NTT ( const int n, int* A, const int tp ) {
	static int lstn = -1, rev[MAXN + 5] {};
	if ( lstn ^ n ) {
		int lgn = log ( n ) / log ( 2 ) + 0.5;
		for ( int i = 0; i < n; ++ i ) rev[i] = ( rev[i >> 1] >> 1 ) | ( ( i & 1 ) << lgn >> 1 );
		lstn = n;
	}
	for ( int i = 0; i < n; ++ i ) if ( i < rev[i] ) A[i] ^= A[rev[i]] ^= A[i] ^= A[rev[i]];
	for ( int i = 2, stp = 1; i <= n; i <<= 1, stp <<= 1 ) {
		int w = qkpow ( G, ( MOD - 1 ) / i );
		if ( ! ~ tp ) w = qkpow ( w, MOD - 2 );
		for ( int j = 0; j < n; j += i ) {
			for ( int k = j, r = 1; k < j + stp; ++ k, r = mul ( r, w ) ) {
				int ev = A[k], ov = mul ( r, A[k + stp] );
				A[k] = add ( ev, ov ), A[k + stp] = sub ( ev, ov );
			}
		}
	}
	if ( ! ~ tp ) for ( int i = 0; i < n; ++ i ) A[i] = mul ( A[i], inv[n] );
}

inline void polyDer ( const int n, const int* A, int* R ) {
	for ( int i = 1; i < n; ++ i ) R[i - 1] = mul ( i, A[i] );
	R[n - 1] = 0;
}

inline void polyInt ( const int n, const int* A, int* R ) {
	for ( int i = n - 1; ~ i; -- i ) R[i + 1] = mul ( inv[i + 1], A[i] );
	R[0] = 0;
}

inline void polyInv ( const int n, const int* A, int* R ) {
	static int tmp[MAXN + 5] {};
	if ( n == 1 ) return void ( R[0] = qkpow ( A[0], MOD - 2 ) );
	int len = 1 << adjust ( n << 1 );
	polyInv ( n + 1 >> 1, A, R );
	for ( int i = 0; i < n; ++ i ) tmp[i] = A[i];
	NTT ( len, tmp, 1 ), NTT ( len, R, 1 );
	for ( int i = 0; i < len; ++ i ) R[i] = mul ( sub ( 2, mul ( tmp[i], R[i] ) ), R[i] ), tmp[i] = 0;
	NTT ( len, R, -1 );
	for ( int i = n; i < len; ++ i ) R[i] = 0;
}

inline void polyLn ( const int n, const int* A, int* R ) {
	static int tmp[2][MAXN + 5] {};
	int len = 1 << adjust ( n << 1 );
	polyDer ( n, A, tmp[0] ), polyInv ( n, A, tmp[1] );
	NTT ( len, tmp[0], 1 ), NTT ( len, tmp[1], 1 );
	for ( int i = 0; i < len; ++ i ) tmp[0][i] = mul ( tmp[0][i], tmp[1][i] );
	NTT ( len, tmp[0], -1 ), polyInt ( n, tmp[0], R );
	for ( int i = 0; i < len; ++ i ) tmp[0][i] = tmp[1][i] = 0;
	for ( int i = n; i < len; ++ i ) R[i] = 0;
}

inline void polyExp ( const int n, const int* A, int* R ) {
	static int tmp[MAXN + 5] {};
	if ( n == 1 ) return void ( R[0] = 1 );
	int len = 1 << adjust ( n << 1 );
	polyExp ( n + 1 >> 1, A, R ), polyLn ( n, R, tmp );
	tmp[0] = sub ( A[0] + 1, tmp[0] );
	for ( int i = 1; i < n; ++ i ) tmp[i] = sub ( A[i], tmp[i] );
	NTT ( len, tmp, 1 ), NTT ( len, R, 1 );
	for ( int i = 0; i < len; ++ i ) R[i] = mul ( R[i], tmp[i] ), tmp[i] = 0;
	NTT ( len, R, -1 );
	for ( int i = n; i < len; ++ i ) R[i] = 0;
}

} // namespace Poly.

int main () {
	inv[1] = 1;
	for ( int i = 2; i <= MAXN; ++ i ) inv[i] = mul ( MOD - MOD / i, inv[MOD % i] );
	n = rint (), m = rint ();
	for ( int i = 1; i <= n; ++ i ) ++ buc[rint ()];
	for ( int i = 1; i <= m; ++ i ) {
		if ( ! buc[i] ) continue;
		for ( int j = 1, lim = m / i, t; j <= lim; ++ j ) {
			t = i * j;
			F[t] = add ( F[t], mul ( inv[j], buc[i] ) );
		}
		buc[i] = 0;
	}
	Poly::polyExp ( m + 1, F, G );
	for ( int i = 1; i <= m; ++ i ) printf ( "%d\n", G[i] );
	return 0;
}
posted @   Rainybunny  阅读(110)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示