zoukankan      html  css  js  c++  java
  • 牡牛和牝牛

    牡牛和牝牛

    有n个0或者1,进行全排列,要求任意两个0间至少有k个1,询问其方案数%5000011。
    对于全部数据,对于全部数据,(1≤N≤10^5,0≤K<N)

    解:

    显然为排列组合问题,考虑方向自然为通项与递推方程。

    法一(通项公式):

    首先0决定了1的摆放,其次数据范围支持对0的枚举,于是枚举0的个数,设其x,于是每个间隔至少要有k个1,不妨先构造让其满足条件,需要(x-1)k个1,显然有((x-1)k+xleq nRightarrow [frac{n+k}{k+1}]),于是问题即变成对于剩下的1的插入0之间的间隔,
    共有x+1个间隔,剩下的1的个数为(n-(x-1)k-x)

    思路一:

    间隔是有序的,而1是无序的,不好解决无序的元素放入有序的盒子的方案数的问题,反过来看则是有序的间隔放入无序的1,而间隔可以多次放入同一个1,即可重组合,于是方案数不难得知为(C_{n-(x-1)k}^{x})


    思路二:

    插入组合问题很困难,于是考虑组合转排列,转换模型,即等价于(n-(x-1)k-x)个1与(x)个0进行全排列,根据可重排列公式有:

    [frac{(n-(x-1)k-x+x)!}{x!(n-(x-1)k-x)!}=frac{(n-(x-1)k)!}{x!(n-(x-1)k-x)!} ]

    [=C_{n-(x-1)k}^{x} ]


    所以只要枚举0的个数,代入公式累加即可。

    参考代码:
    #include <iostream>
    #include <cstdio>
    #define il inline
    #define ri register
    #define ll long long
    #define yyb 5000011
    using namespace std;
    ll jc[100001],jv[100001],lsy(1);
    il ll pow(ll,ll),c(int,int);
    int main(){
        int n,i,j,k,li;scanf("%d%d",&n,&k);
        for(i=jc[0]=1;i<=n;++i)jc[i]=jc[i-1]*i%yyb;
        jv[n]=pow(jc[n],yyb-2),li=(n+k)/(k+1);
        for(i=n,jv[0]=1;i>1;--i)jv[i-1]=jv[i]*i%yyb;
        for(i=1;i<=li;++i)(lsy+=c(n-(i-1)*k,i))%=yyb;
        printf("%lld",lsy);
        return 0;
    }
    il ll c(int n,int r){
        if(n<r)return 0;
        return jc[n]*jv[r]%yyb*jv[n-r]%yyb;
    }
    il ll pow(ll x,ll y){
        ll ans(1);
        while(y){
            if(y&1)ans=ans*x%yyb;
            x=x*x%yyb,y>>=1;
        }return ans;
    }
    

    法二(递推方程)

    经验告诉我们以序列长度为状态,于是设(f[i])表示填到第i个位置的方案数,显然策略为填0或者1,填1恒满足累加(f[i-1]),填0导致前k个都不能填0,故累加(f[i-k-1]),于是有:

    [f[i]=f[i-1]+f[i-k-1] ]

    参考代码:
    #include <iostream>
    #include <cstdio>
    #define il inline
    #define ri register
    #define yyb 5000011
    using namespace std;
    int dp[100001];
    int main(){
        int n,k,i;
        scanf("%d%d",&n,&k);
        for(i=1;i<=k+1;++i)dp[i]=i+1;
        for(i=k+2;i<=n;++i)
            dp[i]=(dp[i-k-1]+dp[i-1])%yyb;
        printf("%d",dp[n]);
        return 0;
    }
    
    
  • 相关阅读:
    守护线程与普通线程
    Java中实现多线程的两种方式之间的区别
    Python os.dup() 方法
    Python os.closerange() 方法
    Python os.close() 方法
    Python os.chroot() 方法
    Python os.chmod() 方法
    JavaScript——基本包装类型
    javascript——function类型(this关键字)
    常用的正则
  • 原文地址:https://www.cnblogs.com/a1b3c7d9/p/10780306.html
Copyright © 2011-2022 走看看