zoukankan      html  css  js  c++  java
  • hdu 5184 类卡特兰数+逆元

    BC # 32 1003

    题意:定义了括号的合法排列方式,给出一个排列的前一段,问能组成多少种合法的排列。

    这道题和鹏神研究卡特兰数的推导和在这题中的结论式的推导:

    首先就是如何理解从题意演变到卡特兰数:

    排列的总长度为 n ,左右括号各为 m = n / 2 个。当给定的排列方式完全合法的时候,剩下需要排列的左右括号的数量就已经确定了,而在排列的过程中,左括号要始终大于等于右括号的数量。设现在有 a 个左括号, b 个右括号,那么这个就可以当做从( a , b )点到 ( m , m )点且不越过直线 y = x 的种类数。将图像下移,则可认为是从( 0 , 0 )点到 ( p , q )点且不越过直线 y = x 的种类数。

    求种类数则是用总的种类数减去非法种类数。总的种类数可用组合数学 C(p+q,p),而非法种类数则是通过图形移动,将图形向下移动一个单位,原本要求越过 y = x ,移动后只要从( 0 , - 1 )点到( p , q - 1 )点且经过 y = x 的路径都是非法的。而计算这个种类数则可以用( 0 , - 1 )与( - 1 , 0 )关于 y = x 对称得,只要从计算从( - 1 , 0 )点到( p , q - 1 )的种类数即可,因为这两点分别在 y = x 的上下,所以路径一定经过 y = x ,种类数为 C ( p + q ,p - 1 );

    相减得到 (( q - p + 1 ) / ( q + 1 )) * C(p+q,p);

    但事实上,直接这样计算是会超时的,因此可以将这个式子继续化成:

    (( p + q )!*( q - p + 1 ))/((q + 1)!* p!)

    在这个式子中大量用到阶乘就是为了可以直接在开始时预处理出阶乘来缩短时间。

    而除法则用到了求逆元。

     1 #include<stdio.h>
     2 #include<string.h>
     3 #define ll long long
     4 const ll mod=1000000007;
     5 char s[1000005];
     6 ll A[1000005];
     7 
     8 void fun(){
     9     A[0]=A[1]=1;
    10     for(int i=2;i<=1000000;i++){
    11         A[i]=(A[i-1]*i)%mod;
    12     }
    13 }
    14 /*
    15 ll C(ll a,ll b){
    16     ll i,ans=1;
    17     for(i=1;i<=b;i++){
    18         ans*=a-i+1;
    19         ans/=i;
    20     }
    21     return ans;
    22 }
    23 */
    24 ll QuickPow(ll a,ll n){
    25     ll tmp=a,ans=1;
    26     tmp %= mod;
    27     while(n){
    28         if(n&1) ans=ans*tmp%mod;
    29         tmp=tmp*tmp%mod;
    30         n>>=1;
    31     }
    32     return ans;
    33 }
    34 
    35 int main(){
    36     int n;
    37     fun();
    38     while(scanf("%d%s",&n,s)!=EOF){
    39         if(n%2)printf("0
    ");
    40         else{
    41             int p=0,q=0,i,l=strlen(s);
    42             for(i=0;i<l;i++){
    43                 if(s[i]=='(')p++;
    44                 else q++;
    45                 if(p<q){
    46                     printf("0
    ");
    47                     break;
    48                 }
    49             }
    50             if(p<q) continue;
    51             {
    52                 p=n/2-p;
    53                 q=n/2-q;
    54                 if(p<0||q<0){
    55                     printf("0
    ");
    56                     continue;
    57                 }
    58                 ll r1=QuickPow(A[p],mod-2),r2=QuickPow(A[q+1],mod-2);
    59                 ll ans=A[p+q];
    60             //    printf("%lld %lld %lld
    ",ans,r1,r2);
    61                 ans=(ans*((q-p+1)%mod))%mod;
    62                 ans=(ans*r1)%mod;
    63                 ans=(ans*r2)%mod;
    64                 printf("%I64d
    ",ans);
    65             
    66             }
    67         }
    68     }
    69     return 0;
    70 }
    View Code
  • 相关阅读:
    Linux系统调用
    Linux的中断 & 中断和异常的区别
    system v和posix的共享内存对比 & 共享内存位置
    epoll里面mmap释疑
    提高网络效率的总结
    推荐相关学习 & 典型算法、典型特征、典型推荐系统框架
    最大似然法理解
    调试多线程 & 查死锁的bug & gcore命令 & gdb对多线程的调试 & gcore & pstack & 调试常用命令
    内存屏障 & Memory barrier
    锁相关知识 & mutex怎么实现的 & spinlock怎么用的 & 怎样避免死锁 & 内核同步机制 & 读写锁
  • 原文地址:https://www.cnblogs.com/cenariusxz/p/4331271.html
Copyright © 2011-2022 走看看