• bzoj 2339: [HNOI2011]卡农


    Description

    题面

    Solution

    比较难想....
    我们先考虑去掉无序的这个条件,改为有序,最后除 (m!) 即可
    (f[i]) 表示前(i)个合法集合的方案数
    明确一点:
    如果前(i-1)个集合已经确定,并且前(i)个是合法的,那么第(i)就是确定的,所以是一一对应的关系,如果不考虑重复和空集的情况,那么总方案数就是 (A_{2^{n}-1}^{i-1})
    考虑去掉不合法的:
    1.当前集合为空集,方案数为 (f[i-1])
    2.有两个集合相同,那么去掉这两个集合的方案数是 (f[i-2]),由于重复的那个位置可以取 (i-1) 个位置,且只与这个集合重复,而不与其他集合重复,所以两个集合可以取 (2^{n}-1-(i-2))
    然后直接递推即可,(log)求逆的话,复杂度就是 (O(n*logn))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    const int N=1e6+5,mod=1e8+7;
    int qm(int x,int k){
      int sum=1;
      while(k){
        if(k&1)sum=1ll*sum*x%mod;
        x=1ll*x*x%mod;k>>=1;
      }
      return sum;
    }
    int f[N],Fac[N];
    int main(){
      freopen("pp.in","r",stdin);
      freopen("pp.out","w",stdout);
      int n,m,x,t,jc=1;
      cin>>n>>m;
      t=qm(2,n);Fac[1]=x=(t-1+mod)%mod;
      Fac[0]=1;
      for(int i=1;i<=m;i++)
        jc=1ll*i*jc%mod,Fac[i]=1ll*Fac[i-1]*x%mod,x=(x-1+mod)%mod;
      f[0]=1;f[1]=0;
      for(int i=2;i<=m;i++){
        f[i]=Fac[i-1];
        f[i]=(f[i]-f[i-1]-1ll*f[i-2]*(i-1)%mod*(t-1-(i-2)))%mod;
      }
      if(f[m]<0)f[m]+=mod;
      f[m]=1ll*f[m]*qm(jc,mod-2)%mod;
      cout<<f[m]<<endl;
      return 0;
    }
    
    
  • 相关阅读:
    (二)php的常量和变量
    关于标签系统的一点想法。
    Linux运维工程师中级面试题
    Linux C 编程内存泄露检测工具(一):mtrace
    掌握sudo的使用
    Scala极速入门
    处理千万级以上的数据提高查询速度的方法
    linux svn服务器搭建、客户端操作、备份与恢复
    select/poll/epoll 对比
    汇编指令和标志寄存器
  • 原文地址:https://www.cnblogs.com/Yuzao/p/8445693.html
Copyright © 2020-2023  润新知