- 题意:有n种调料包,每种调料包要放在至少两个碗里,每个碗的调料状态都要不同。
- 思路:想了一个下午都没有成功,问了Leasier同学才知道。
她说的是二项式反演,但实际上本质就容斥。
ans=∑ n i=1(−1)i∗f[i]∗Cin
f[i]表示至少有i种调料个数小于2的方案书
因为乘了Cin所以我们直接可以把前i位当做坏调料,i以后的是随便放的。
考虑到性质:这i种调料放入的碗数j小于等于i
所以搞一个类似第二类斯特林数的 g[i][j]:前i种调料放入j个碗的方案数。
g[i][j]=g[i−1][j−1]+g[i−1][j]∗(j+1)
j+1相当于在之前j个碗种选一个或者不放。
f[i]=∑ i j=1g[i][j]∗2(n−i)∗j∗22n−i
然后就over啦!
ps(补充).注意指数的模数是m−1因为扩展欧拉定理
- 黛玛
#include<bits/stdc++.h>
using namespace std;
const int N=3005;
typedef long long ll;
ll f[N],C[N][N],n,m,g[N][N],p2[N*N],p22[N*N],p4[N*N];
ll ksm(ll a,ll b) {
ll res=1;
for(;b;b>>=1,a=a*a%m) if(b&1)res=res*a%m;
return res;
}
void init() {
for(int i=0;i<=n;i++) {
g[i][0]=1;
for(int j=1;j<=i;j++) {
g[i][j]=(g[i-1][j-1]+g[i-1][j]*(j+1)%m)%m;
}
}
p2[0]=p22[0]=1;
for(int i=1;i<=n*n;i++) p2[i]=p2[i-1]*2%m,p22[i]=p22[i-1]*2%(m-1);
f[0]=0;
for(int i=0;i<=n;i++) {
for(int j=0;j<=i;j++) {
f[i]+=g[i][j]*p2[(n-i)*j]%m,f[i]%=m;
}
f[i]=f[i]*ksm(2,p22[n-i])%m;
}
for(int i=0;i<=n;i++) C[i][0]=1;
for(int i=1;i<=n;i++) {
for(int j=1;j<=i;j++) {
C[i][j]=(C[i-1][j-1]+C[i-1][j])%m;
}
}
}
int main() {
scanf("%lld%lld",&n,&m);
init();
ll ans=0;
for(ll j=1,i=0;i<=n;i++) {
if(i&1) f[i]*=-1;
ans=(ans+C[n][i]%m*f[i]%m)%m;
}
printf("%lld",(ans+m)%m);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· winform 绘制太阳,地球,月球 运作规律
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人