noip模拟 *置换

暴力就是枚举每种情况,算 \(lcm\)
然后仔细想想(虽然考场没想)就会发现,最后图会形成若干个环
我们要求的其实就是环大小的 \(lcm\)
根据这个性质,可以得到一个 \(60\) 分的 \(dp\)
\(dp_{i,j}\) 表示一共选了 \(i\) 个数,\(lcm\)\(j\) 的方案数
我们枚举环的大小 \(s\) 和数量 \(a\),可以转移

\[dp_{i+s*a,lcm(j,s)}=dp_{i,j}*\prod_{z=1}^{a}\binom{i+s*z}{s}(a!)^{-1}*((s-1)!)^{a} \]

转移的第二项是在求选数的方案,最好自己理解一下
第三项是求单位环连边的方案

正解

\(lcm\) 过大(状态数太大),在写 \(60\) 分的时候还要写 \(vec\)
考虑怎么优化
把一个 \(lcm\) 拆成若干质因子,发现大于 \(\sqrt{n}\) 的质因子出现次数不会大于 \(1\)
那我们把所有小于 \(\sqrt{n}\) 的质因子可以构成的 \(lcm\) 处理出来,发现数量不超过 \(2000\)
那这道题就可以做了
处理出每个数被小于 \(\sqrt{n}\) 的质因子处理尽的值(设其为 \(t\) ),把 \(t*t\) 乘进方案就好了
因为 \((ab)^2=a^2b^2\)

Code
#include <bits/stdc++.h>
#define re register
#define db double
#define int long long
#define ll long long
#define pir make_pair
#define memset(a) { for(re int i=1,lim=2*n;i<=lim;i++) for(re int j=1,lim=2*n;j<=lim;j++) a[i][j]=0; }
using namespace std;
const int maxn=5e3+10;
const int INF=1e9+10;
// const int mol=998244353;
inline int qpow(int a,int b,int mol) { int ans=1; while(b) { if(b&1) (ans*=a)%=mol; (a*=a)%=mol; b>>=1; } return ans; }
inline int read(){
    int x=0,w=1;char ch=getchar();
    while(ch<'0'||ch>'9') w=(ch=='-')?-1:1,ch=getchar();
    while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
    return x*w;
}

int prim[10]={0,2,3,5,7,11,13,17,19};
int n,mol,a[maxn],g[520][maxn],f[520][maxn],fac[maxn],inv[maxn];
inline int C(int n,int m) { return fac[n]*inv[n-m]%mol*inv[m]%mol; }
inline int lcm(int a,int b) { return a*b/__gcd(a,b); }
int ks[maxn],tot; int pre[maxn],dik[maxn];
inline void dfs(int num,int sum,int x) { 
    if(num>n) { return; }
    if(x>6) { ks[++tot]=sum; return; }
    int tmp=1;
    dfs(num,sum,x+1);
    for(re int i=1;i<=n;i++) {
        tmp*=prim[x];
        if(num+tmp>n) break;
        dfs(num+tmp,sum*tmp,x+1);
    }
}
signed main(void) {
    // freopen("erp.in","r",stdin);
    freopen("perm.in","r",stdin); freopen("perm.out","w",stdout); 
    n=read(); mol=read();
    fac[0]=1; for(re int i=1;i<=n;i++) fac[i]=fac[i-1]*i%mol;
    inv[n]=qpow(fac[n],mol-2,mol); for(re int i=n;i>=1;i--) inv[i-1]=inv[i]*i%mol;
    dfs(0,1,1); sort(ks+1,ks+tot+1);
    for(re int i=1;i<=n;i++) {
        pre[i]=dik[i]=i;
        for(re int j=1;j<=6;j++) while(dik[i]%prim[j]==0) dik[i]/=prim[j];
    }
    sort(pre+1,pre+1+n,[](int a,int b){return dik[a]<dik[b];});
    f[0][1]=1;
    for(re int i=1,d=i;i<=n;i=d) {
        for(re int j=0;j<=n;j++) for(re int k=1;k<=tot;k++) g[j][k]=f[j][k];
        while(dik[pre[i]]==dik[pre[d]]) {
            int ls=pre[d]; d++;
            for(re int j=tot;j>=1;j--) {
                int lc=lcm(ks[j],ls/dik[ls]);
                if(lc>ks[tot]) continue;
                lc=lower_bound(ks+1,ks+1+tot,lc)-ks;
                for(re int a=n-ls;a>=0;a--) if(g[a][j]) {
                    int ds=1;
                        for(re int z=1;z*ls+a<=n;z++) {
                            (ds*=C(z*ls+a,ls))%=mol;
                            (g[z*ls+a][lc]+=g[a][j]*ds%mol*qpow(fac[ls-1],z,mol)%mol*inv[z]%mol)%=mol;
                        }
                }
            }
        }
        for(re int j=0;j<=n;j++) for(re int k=1;k<=tot;k++) (f[j][k]+=(g[j][k]-f[j][k])*dik[pre[i]]%mol*dik[pre[i]]%mol)%=mol;
    }
    int ans=0;
    for(re int i=1;i<=tot;i++) (ans+=f[n][i]*ks[i]%mol*ks[i]%mol)%=mol;
    (ans*=inv[n])%=mol;
    printf("%lld\n",ans);
}
posted @ 2021-10-01 19:33  zJx-Lm  阅读(25)  评论(0编辑  收藏  举报