Luogu P2727 【01串 Stringsobits】

看到题解里好像都是用$DP$解决的,本着禁止DP的原则,我来提供一发纯数学其实和DP本质相同的题解,前两天刚反演题,脑子炸了,本来说换换脑子,结果还是数学

首先受进制思想启发,我们不妨按位考虑,考虑这一位选一对排列编号造成的影响——即让整个数的编号向后推移了多少

容易想到,这一位选一,编号增加了之后几位满足条件任选的方案数,即第$i$位选一,$cnt$表示前几位选了几个一

$$id+=sum_{j=0}^{min(i-1,L-cnt)}calc(i-1,j)$$

$clac(x,y)$表示前面$y$位,选$x$位为一的方案数,这个就是一个可重集排列问题,即

$$clac(x,y)=frac{y!}{x!*(y-x)!}$$

因为$n!$太大会爆$long~long$,所以我们可以使用唯一素数分解定理把阶乘拆成质因子的乘积,然后再乘起来

上代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#define int long long
using namespace std;
int pr[10]={2,3,5,7,11,13,17,19,23,29};
int n,k,rk,cnt,ans[50],cp[20];
void add(int x,int c)
{
    //唯一素数分解
    for(int i=1;i<=x;i++)
        for(int tmp=i,j=0;j<10&&tmp>1;j++)
            while(tmp%pr[j]==0)
                tmp/=pr[j],cp[j]+=c;
}
int make(int x,int y)
{
    //可重集排列
    int ret=1;
    memset(cp,0,sizeof(cp));
    add(x,1),add(y,-1),add(x-y,-1);
    for(int i=0;i<10;i++)
        for(int j=1;j<=cp[i];j++)
            ret*=pr[i];
    return ret;
}
signed main()
{
    scanf("%lld%lld%lld",&n,&k,&rk);
    rk--;    //因为有=0的情况,所以rk-1
    if(!rk)
    {
        for(int i=1;i<=n;i++)
            printf("0");
        printf("
");
        return 0;
    }
    for(int i=n;i;i--)
    {
        //按位考虑选或不选
        int sum=0;
        for(int j=0;j<=min(i-1,k-cnt);j++)
            sum+=make(i-1,max(j,i-1-j));
        if(rk>=sum)
            rk-=sum,ans[i]=1,cnt++;
    }
    for(int i=n;i;i--)
        printf("%lld",ans[i]);
    printf("
");
    return 0;
}
原文地址:https://www.cnblogs.com/ivanovcraft/p/11341571.html