卡特兰数:

1 通项公式:h(n)=C(n,2n)/(n+1)=(2n)!/((n!)*(n+1)!)

2递推公式:h(n)=((4*n-2)/(n+1))*h(n-1); h(n)=h(0)*h(n-1)+h(1)*h(n-2)+...+h(n-1)*h(0).

3前几项为:h(0)=1,h(1)=1,h(2)=2,h(3)=5,h(4)=14,h(5)=42,......

4应用场景:

a.括号化问题。
  矩阵链乘: P=a1×a2×a3×……×an,依据乘法结合律,不改变其顺序,只用括号表示成对的乘积,试问有几种括号化的方案?(h(n)种)
b.出栈次序问题。
  一个栈(无穷大)的进栈序列为1,2,3,..n,有多少个不同的出栈序列?
  类似:
  (1)有2n个人排成一行进入剧场。入场费5元。其中只有n个人有一张5元钞票,另外n人只有10元钞票,剧院无其它钞票,问有多少中方法使得只要有10元的人

买 票,售票处就有5元的钞票找零?(将持5元者到达视作将5元入栈,持10元者到达视作使栈中某5元出栈)
  (2)在圆上选择2n个点,将这些点成对连接起来,使得所得到的n条线段不相交的方法数。

c.将多边行划分为三角形问题。
  (1)将一个凸多边形区域分成三角形区域的方法数?
  (2)类似:一位大城市的律师在她住所以北n个街区和以东n个街区处工作。每天她走2n个街区去上班。如果她从不穿越(但可以碰到)从家到办公室的对角线,那            么有多少条可能的道路?
  (3)类似:在圆上选择2n个点,将这些点成对连接起来使得所得到的n条线段不相交的方法数?
d.给顶节点组成二叉树的问题。
  给定N个节点,能构成多少种形状不同的二叉树?
  (一定是二叉树!先去一个点作为顶点,然后左边依次可以取0至N-1个相对应的,右边是N-1到0个,两两配对相乘,就是

h(0)*h(n-1) + h(2)*h(n-2) +  + h(n-1)h(0)=h(n))(能构成h(N)个)。

下面给出hdu上面的有关例题:

hdoj 1134

2n个人围成一个圆圈,求两两相互握手并且不交叉的所有握手方式。
这个是卡特兰数的一个例子,设2n个人一共有h(n)种,那么现在第一个人可以和第2,4,6,。。。,2(n-1),2n,即必须保证和他握手的那个人两边是偶数,即为:
h(n)=h(0)*h(n-1)+h(1)*h(n-2)+...+h(n-1)*h0=(4*n-2)/(n+1) *h(n-1),h(0)=1,h(1)=1.
通项公式:h(n)=C(n,2n)/n+1=(2n)!/((n!)*(n+1)!)
但是这个题目是大数,所以必须采用数组模拟乘除法.

源码:

#include <iostream>
#include <stdio.h>
using namespace std;
const int N=105;
int catalan[102][N];
void setCatalan()
{memset(catalan,0,sizeof(catalan));catalan[1][0]=1;int i,tmp[N],j,yushu,m;for(i=2;i<=100;i++){m=4*i-2;//大数乘法for(j=0;j<N;j++){catalan[i][j]+=catalan[i-1][j]*m;if(catalan[i][j]>=10){catalan[i][j+1]+=catalan[i][j]/10;catalan[i][j]=catalan[i][j]%10;}}for(j=0;j<N;j++)tmp[j]=0;yushu=0;m=i+1;//大数/小数for(j=N-1;j>=0;--j){tmp[j]=(10*yushu+catalan[i][j])/m;yushu=(10*yushu+catalan[i][j])%m;}for(j=0;j<N;++j)catalan[i][j]=tmp[j];}
}
int main()
{setCatalan();int i,n;while(cin>>n && (-1 != n)){i=N-1;while(!catalan[n][i])--i;for(;i>=0;--i)cout<<catalan[n][i];cout<<endl;}return 0;
}

hdoj1023

求出栈序列,比如1,2,3,出栈序列为3 2 1,1 2 3,1 3 2,2 1 3,2 3 1,一共5种
我们把入栈看做1,出栈看做0,那么入栈出栈看做一系列的1010。。。,但是必须保证从左往右
看的时候1必须多余0,这个是卡塔兰数的第二个应用,种数为:C(n,2n)-C(n+1,2n).
粗略这样理解:我们从2n个位置中选出n个来存放1,方法数为C(n,2n),减去不满足的情况。
不合法的情况:我们在2n个位置放n+1个0,n-1个1,由于0的个数多2个,2n为偶数,故必在某一个奇数位上出现0的累计数超过1的累计数。同样在后面部分0和1互换,使之成为由n个0和n个1组成的2n位数,即n+1个0和n-1个1组成的2n位数必对应一个不符合要求的数,即C(n+1,2n)。
h(n)=C(n,2n)-C(n+1,2n).

hdoj1130

给出n个点,求组成二叉树的所有种数,2个点组成2种二叉树,3个点组成5种二叉树。。。
这个也是卡塔兰数的一个应用,和1134类似,我们去除一个点作为根节点,然后左边依次可以取0至N-1个相对应的,右边是N-1到0个,两两配对相乘,就是h(0)*h(n-1) + h(2)*h(n-2) +  + h(n-1)h(0)=h(n))。

hdoj2067

给出一个棋盘n*n,求从左下角到右上角的不经过对角线的所有走法,这个经过分析也是卡特兰数。我们把往右走看做1,把往上走看做0,那么从左向右看做一系列的101100.。。,和那个求出栈序列的就是一个问题了,即0的个数不能超过1,由于上半角和下半角一样,所以求出来卡特兰数*2就是我们的答案了。

hdoj1133
买票问题:有m个人手里拿的是50元的,n个人拿的是100元的,问使买票过程不中断的排队方式。我们知道如果前面出现拿50的人小于拿100的人,那么肯定出现找不开的情况,我们把拿50的看做0,拿100的看做1,所以从左往右看的时候0的个数必须大于1。我们知道总的情况为
C(n,m+n),需要求出不合法的序列个数,还是之前的思路,存在一个奇数位置2*k+1,使得0出现k此1出现k+1次,后面会有(m-k)个0,(n-k-1)个1,我们将01交换,即这个序列共有m+1个1,n-1个0,这个序列的所有排序情况就对应了一种不合法的序列情况(可以这样理解:由于m+1>n-1,那么必然在某一个位置出现1的个数大于0的个数,这样在这个位置往后的01我们交换回来,就对于了一种不合法的序列了),即C(m+1,m+n),最后的结果为:ans=C(n,m+n)-C(m+1,m+n),其中(m>=n);当m<n时买票过程必然中断。

#include <iostream>
using namespace std;
const int MAX=400;
const int BASE=1000;
int catalan[MAX],tmp[MAX];
int main()
{int i,j,n,m,cas=0;while(cin>>m>>n){if(!(m+n))break;if(m<n)printf("Test #%d:\n0\n",++cas); else{memset(catalan,0,sizeof(catalan));memset(tmp,0,sizeof(tmp));catalan[0]=1;//(m+n)!for(i=1;i<=(m+n);++i){for(j=0;j<MAX;j++)catalan[j]=catalan[j]*i;for(j=0;j<MAX;j++){if(catalan[j]>=BASE){catalan[j+1]+=catalan[j]/BASE;catalan[j]=catalan[j]%BASE;}}}//*(m-n+1)int k=m-n+1;for(j=0;j<MAX;j++)catalan[j]=catalan[j]*k;for(j=0;j<MAX;j++){if(catalan[j]>=BASE){catalan[j+1]+=catalan[j]/BASE;catalan[j]=catalan[j]%BASE;}}// /(m+1)int yushu=0;k=m+1;for(j=MAX-1;j>=0;--j){tmp[j]=(BASE*yushu+catalan[j])/k;yushu=(BASE*yushu+catalan[j])%k;}printf("Test #%d:\n",++cas);j=MAX-1;while(!tmp[j])--j;//   cout<<"j"<<j<<endl;printf("%d",tmp[j]);--j;for(;j>=0;--j)printf("%03d",tmp[j]);printf("\n");}}return 0;
}

转载于:https://www.cnblogs.com/buptLizer/archive/2011/10/23/2222027.html

数论二(hdoj 卡特兰数)相关推荐

  1. 【数论】——Catalan 卡特兰数

    Catalan 卡特兰数 文章目录 Catalan 卡特兰数 定义 计算 证明 推论 代码 定义 给定 n 个 0 和 n 个 ** 1**,它们将按照某种顺序排成长度为 **2n ** 的序列,求它 ...

  2. 数据结构 | 算法中的卡特兰数的应用

    卡特兰数又称卡塔兰数,卡特兰数是组合数学中一个常出现在各种计数问题中出现的数列. 一.简单介绍 输入一个整数n,计算h(n). 令h(0)=1,h(1)=1, Catalan数满足递推式:h(n)= ...

  3. [编程之美]买票找零(卡特兰数)

    第一次看这题的时候没有好好注意,后来发现这是一类大问题,学习了卡特兰数这个概念,顺便又复习了高中的排列组合知识... 一.书中问题 先看一下书中引入卡特兰数的例子: <编程之美>4.3买票 ...

  4. HDOJ/HDU 1133 Buy the Ticket(数论~卡特兰数~大数~)

    Problem Description The "Harry Potter and the Goblet of Fire" will be on show in the next ...

  5. 面试题--5个数能组成多少中不同的二叉搜索树的结构--卡特兰数

    面试题--5个数能组成多少中不同的二叉搜索树的结构--卡特兰数 题目描述: 1,2,3,4,5 五个数字,能组成多少种不同的二叉搜索树的结构?(D) A. 32 B. 36 C. 40 D. 42 解 ...

  6. 算法基础 - 数论 | 组合数学 卡特兰数(Catalan number)定义、证明及例题

    写在前面:卡特兰数这东西感觉挺常用的,并且公式很简单,那就花一下午总结一下,学点皮毛吧(反正遇到我还是不会 ) [PDF] 大三上组合数学课堂讲义 文章目录 卡特兰数定义 卡特兰数的性质 卡特兰数证明 ...

  7. HDU 3240 Counting Binary Trees 数论-卡特兰数

    题目地址: http://acm.hdu.edu.cn/showproblem.php?pid=3240 卡特兰数递推公式h(i)=h(i-1)*(4*i-2)/(i+1) 如果直接算每一步,然后mo ...

  8. HDOJ 5184 Brackets 卡特兰数扩展

    既求从点(0,0)仅仅能向上或者向右而且不穿越y=x到达点(a,b)有多少总走法... 有公式: C(a+b,min(a,b))-C(a+b,min(a,b)-1)  /// 折纸法证明卡特兰数: h ...

  9. [超级码力在线编程大赛初赛(二)] 4.小栖的金字塔 施罗德数(超级卡特兰数)

    题目链接:小栖的金字塔 题意 小栖可以在不同点间移动,假设小栖现在在(x1,y1){(x_1,y_1)}(x1​,y1​),他能够移动到的下一个点(x2,y2){(x_2,y_2)}(x2​,y2​) ...

最新文章

  1. HttpRequest接口测试
  2. 导入Excel和CSV文件
  3. flutter嵌入HTML5页面,Flutter加载Html并实现与JS 的双向调用
  4. Linux零基础初级教程
  5. 在OSGi中为Karaf构建Camel-CXF REST服务–组播和聚合
  6. 小学计算机教学教师培训,例谈小学信息技术课堂的有效教学
  7. 算命数据_未来的数据科学家或算命精神向导
  8. 利用一维数组输出杨辉三角
  9. 研究生导师一般希望招什么样的研究生?
  10. Java经典编程题,你学废了吗?
  11. 2021年顶级服务器备份软件和解决方案
  12. Android 虚拟多开系列二——技术原理
  13. 【C++】给定两个没有刻度的容器,对于任意给定的容积,求出如何只用两个瓶装出L升的水
  14. matlab打反斜杠,[转载]转义字符 反斜杠
  15. 硬件模拟大师_用机器人做咖啡,“智咖大师”这样升级新零售
  16. react-native 关闭黄色警告
  17. Java破解9X9数独小游戏
  18. 蒙特卡洛模拟电动汽车充电matlab,基于蒙特卡洛模拟的电动汽车充电负荷预测
  19. spark-sql总结
  20. [LeetCode]Buy and Sell Stocks 买卖股票问题

热门文章

  1. java模拟dos窗口
  2. 在不重装系统的情况下创建Linux的Swap分区
  3. openwrt顶层Makefile分析-转
  4. L1-078 吉老师的回归 (15 分)-PAT 团体程序设计天梯赛 GPLT
  5. 蓝桥杯 ALGO-65 算法训练 比赛安排
  6. 蓝桥杯 ALGO-99 算法训练 薪水计算
  7. jQuery LigerUI API预览版发布
  8. perl 包下载官方网站(速度很快的) 和 解压安装指令
  9. Java Servlet 和JSP教程(2)
  10. Java基础系列—List排序