Add or Multiply 1 题解(第二类斯特林数)

题目链接

题目思路

这个题目

首先一个算式肯定是加法和乘法交替的。对于每组连续的加法或者乘法,如果组内的元素是相同的,那么运算的结果也是一样的。假设分成了 \(x\)组加法和\(y\)组乘法,满足 \(|x-y|\leq 1\),那么加法和乘法的分组都是独立的。问题等价于将 \(n\)个有标号的球分到 \(m\) 个有标号的集合里面,使得每个集合都是非空的。这个问题可以用第二类斯特灵数简单解决。

然后直接枚举n分为了\(i\)组,那么m可能分为\(i-1,i,i+1\)组相乘即可

但是对于第\(m\)分为\(i\)组要乘以\(2\)

因为根据隔板法第一组可能为乘法或者加法

代码

#include<bits/stdc++.h>
#define fi first
#define se second
#define debug printf(" I am here\n");
using namespace std;
#define S dp
typedef long long ll;
typedef unsigned long long ull;
typedef pair<ll,ll> pii;
mt19937 rnd(time(0));
const ll INF=0x3f3f3f3f3f3f3f3f;
const int maxn=3e3+5,inf=0x3f3f3f3f,mod=1e9+7;
const double eps=1e-10;
int n,m;
ll dp[maxn][maxn];
ll fac[maxn];
signed main(){
        for(int i=0;i<=3000;i++){
        if(i==0){
            fac[i]=1;
        }else{
            fac[i]=fac[i-1]*i%mod;
        }
    }
    dp[0][0]=1;
    for(int i=1;i<=3000;i++){
        for(int j=1;j<=i;j++){
            dp[i][j]=(dp[i-1][j-1]+j*dp[i-1][j])%mod;
        }
    }
    for(int i=1;i<=3000;i++){
        for(int j=1;j<=i;j++){
            dp[i][j]=dp[i][j]*fac[j]%mod;
        }
    }
    int _;scanf("%d",&_);
    while(_--){
        scanf("%d%d",&n,&m);
        ll ans=0;
        for(int i=1;i<=n;i++){
            ll temp=(dp[m][i-1]+2*dp[m][i]+dp[m][i+1])%mod;
            ans=(ans+temp*dp[n][i])%mod;
        }
        printf("%lld\n",ans);
    }
    return 0;
}

posted @ 2021-08-23 10:35  hunxuewangzi  阅读(45)  评论(0编辑  收藏  举报