zoukankan      html  css  js  c++  java
  • hdu 6432 Cyclic

    题意:

    计算长度为n且不含有子串[i,i+1]或[n,1]的不同循环排列的个数。

    解法:

    拿n=4举例,即不含12,23,34,41这样的子串。

    首先先说明什么是循环排列:

    即把1-n这n个数随意地放到一个圆圈上,循环排列的不同仅仅取决于这n个数的相对位置的不同。

    例如1234,2341,3412,4123这些数为相同的循环排列数。

    循环排列没有首末之分,这四个元素随便从哪一个元素开始,绕一个方向转过去,都不改变它们的相对顺序;直线排列则首末分明,原来排末位,调换排首位,已改变它们的相对顺序。循环排列与直线排列的主要区别就在这一点上。

    从例子看出,直线排列的个数是循环排列个数的n倍

    由直线排列个数为n!可推知循环排列个数为(n-1)!。

    讲完了循环排列,再来看此题是求不含子串[i,i+1]或[n,1](以下简称顺序子串,共有n个)的循环排列个数

    因为一个排列中可能含有多个顺序子串,所以我们列举至少含有0个,1个,...n个的情况  (注意是至少,因为无法保证恰好含有i个)

    包含至少一个顺序子串的循环排列数为C(n,1)*(n-2)!

    包含至少两个顺序子串的循环排列数为C(n,2)*(n-3)!

    ...

    包含至少k个顺序子串的循环排列数为C(n,k)*(n-k-1)! 

    (为什么是(n-k-1)! 当你选出了k个子串之后,至少有k+1个数相对位置已被确定,我们让剩下的(n-k-1)个数全排列即可。)

    同时注意到包含n个顺序子串的循环排列数一定是1个。

    事件之间相互包含,所以用到容斥原理:

     ∑(k从0到n-1)(-1)^k*C(n,k)*(n-k-1)!+(-1)^n*1

    同样预处理阶乘及阶乘的逆元。

    AC代码:

     1 #include <bits/stdc++.h>
     2 #define ll long long
     3 using namespace std;
     4 const int mod=998244353;
     5 const int maxn=1e5+10;
     6 ll fac[maxn],invfac[maxn],inv[maxn];
     7 void init()
     8 {
     9     int i;
    10     inv[1]=1;
    11     for(i=2;i<maxn;i++)
    12     {
    13         inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod;
    14     }
    15     fac[0]=1;
    16     invfac[0]=1;
    17     for(i=1;i<maxn;i++)
    18     {
    19         fac[i]=(ll)fac[i-1]*i%mod;
    20         invfac[i]=(ll)invfac[i-1]*inv[i]%mod;
    21     }
    22 }
    23 ll C(int n,int m)
    24 {
    25     if(n<0||m<0||m>n)return 0;
    26     return (ll)fac[n]*invfac[m]%mod*invfac[n-m]%mod;
    27 }
    28 int main()
    29 {
    30     int t;
    31     cin>>t;
    32     init();
    33     while(t--)
    34     {
    35         int n,k;
    36         scanf("%d",&n);
    37         ll ans=0;
    38         for(k=0;k<=n-1;k++)
    39         {
    40             if(k&1)
    41             {
    42                 ans=(ans-(ll)C(n,k)*fac[n-k-1]%mod+mod)%mod;
    43             }
    44             else ans=(ans+(ll)C(n,k)*fac[n-k-1]%mod)%mod;
    45         }
    46         if(n&1)ans-=1;
    47         else ans+=1;
    48         printf("%I64d
    ",ans);
    49     }
    50     return 0;
    51 }
  • 相关阅读:
    如何理解C语言的左结合 和右结合性
    Egg项目使用vscode的debug模式跑单元测试
    为什么要用MongoDB副本集
    理解JS原型和原型链
    防止重复请求攻击
    引擎、编译器和作用域
    闭包原理解析及其应用场景
    树形结构数据完美解决方案
    Excel文件导入导出(基于Nodejs、exceljs)
    架构层面高并发解决方案选择----项目架构模式选择
  • 原文地址:https://www.cnblogs.com/raincle/p/9521266.html
Copyright © 2011-2022 走看看