卡特兰(Catalan)数入门详解
基本概念
介绍
学卡特兰数我觉得可能比组合数要难一点,因为组合数可以很明确的告诉你那个公式是在干什么,而卡特兰数却像是在用大量例子来解释什么时卡特兰数
这里,我对卡特兰数做一点自己的理解
卡特兰数是一个在组合数学里经常出现的一个数列,它并没有一个具体的意义,却是一个十分常见的数学规律
对卡特兰数的初步理解:有一些操作,这些操作有着一定的限制,如一种操作数不能超过另外一种操作数,或者两种操作不能有交集等,这些操作的合法操作顺序的数量
为了区分组合数,这里用\(f_n\)表示卡特兰数的第\(n\)项
从零开始,卡特兰数的前几项为\(1, 1, 2, 5, 14, 42, 132, 429, 1430, 4862, 16796, 58786, 208012, 742900, 2674440, 9694845, 35357670, 129644790\ldots\)
定义
递归定义
\(f_n=f_0*f_{n-1}+f_1*f{n-2}+\ldots+f_{n-1}f_{0}\),其中\(n\geq 2\)
递推关系
\(f_n=\frac{4n-2}{n+1}f_{n-1}\)
通项公式
\(f_n=\frac{1}{n+1}C_{2n}^{n}\)
经化简后可得
\(f_n=C_{2n}^{n}-C_{2n}^{n-1}\)
只要我们在解决问题时得到了上面的一个关系,那么你就已经解决了这个问题,因为他们都是卡特兰数列
实际问题
先用一个最经典的问题来帮助理解卡特兰数
去掉了所有的掩饰,将问题直接写出来就是
例题1
在一个\(w\times h\)的网格上,你最开始在\(\left(0,0\right)\)上,你每个单位时间可以向上走一格,或者向右走一格,在任意一个时刻,你往右走的次数都不能少于往上走的次数,问走到\(\left(n,m\right),0\leq n\)有多少种不同的合法路径。
合法路径个数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
直接求不好,考虑求有多少种不合法路径
路径总数为在\(2n\)次移动中选\(n\)次向上移动,即\(C_{2n}^{n}\)
画一下图,我们把\(y=x+1\)这条线画出来,发现所有的合法路径都是不能碰到这条线的,碰到即说明是一条不合法路径
先随便画一条碰到这条线的不合法路径,所有的不合法路径都会与这条线有至少一个交点,我们把第一个交点设为\(\left(a,a\right)\)
如图

我们把\(\left(a,a\right)\)之后的路径全部按照\(y=x+1\)这条线对称过去
这样,最后的终点就会变成\(\left(n-1,n+1\right)\)

由于所有的不合法路径一定会与\(y=x+1\)有这么一个交点
我们可以得出,所有不合法路径对称后都唯一对应着一条到\(\left(n-1,n+1\right)\)的路径
且所有到\(\left(n-1,n+1\right)\)的一条路径都唯一对应着一条不合法路径(只需将其对称回去即可)
所以不合法路径总数是\(C_{2n}^{n-1}\)
那么合法的路径总数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
这是一个非常好用且重要的一个方法,其它的问题也可以用该方法解决
即找到不合法路径唯一对应的到另一个点的路径
如网格计数
方法
先将方法写在前面吧
相信大家都听过烧开水这个数学小故事吧
和学习数学一样,转化是基本思路,将一个问题转化为另外一个已经解决了的问题是最重要的
01序列
你现在有\(n\)个\(0\)和\(n\)个\(1\),问有多少个长度为\(2n\)的序列,使得序列的任意一个前缀中\(1\)的个数都大于等于\(0\)的个数
例如\(n=2\)时
有\(1100,1010\)两种合法序列
而\(1001,0101,0110,0011\)都是不合法的序列
合法的序列个数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
我们把出现一个\(1\)看做向右走一格,出现一个\(1\)看做向上走一格,那么这个问题就和上面的例题\(1\)一模一样了
拓展
如果是\(n\)个\(1,m\)个\(0\)呢?
不过是最后的终点变为了\(\left(n,m\right)\)罢了
如果是\(1\)的个数不能够比\(m\)少\(k\)呢
我们只需将\(y=x+1\)这条线上下移动即可
括号匹配
你有\(n\)个左括号,\(n\)个右括号,问有多少个长度为\(2n\)的括号序列使得所有的括号都是合法的
合法的序列个数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
要使所有的括号合法,实际上就是在每一个前缀中左括号的数量都不少于右括号的数量
将左括号看做\(1\),右括号看做\(0\),这题又和上面那题一模一样了
进出栈问题
有一个栈,我们有\(2n\)次操作,\(n\)次进栈,\(n\)次出栈,问有多少中合法的进出栈序列
合法的序列个数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
要使序列合法,在任何一个前缀中进栈次数都不能少于出栈次数\(\ldots\)
后面就不用我说了吧,和上面的问题又是一模一样的了
312排列
一个长度为\(n\)的排列\(a\),只要满足\(i<j<k\)且\(a_j<a_k<a_i\)就称这个排列为\(312\)排列
求\(n\)的全排列中不是\(312\)排列的排列个数
答案也是卡特兰数
我们考虑\(312\)排列有什么样的特征
如果考虑一个排列能否被\(1,2,3,\ldots,n-1,n\)排列用进栈出栈来表示
那么\(312\)排列就是所有不能被表示出来的排列
那么这个问题就被转化成进出栈问题了
不相交弦问题
在一个圆周上分布着 \(2n\)个点,两两配对,并在这两个点之间连一条弦,要求所得的\(2n\)条弦彼此不相交的配对方案数
当\(n=4\)时,一种合法的配对方案为如图

合法的序列个数为\(C_{2n}^{n}-C_{2n}^{n-1}\)
这个问题没有上面的问题那么显然,我们规定一个点为初始点,然后规定一个方向为正方向
如规定最上面那个点为初始点,逆时针方向为正方向
然后我们把一个匹配第一次遇到的点(称为起点)旁边写一个左括号\((\),一个匹配第二次遇到的点(称为终点)旁边写一个右括号\()\)
如图

看出来吗,在规定了这样的一个顺序后,在任意一个前缀中起点的个数都不能少于终点的个数
于是这又是一个卡特兰数列了
二叉树的构成问题
有\(n\)个点,问用这\(n\)个点最终能构成多少二叉树
答案仍然是卡特兰数列
这个问题不是用上面的方法,是用递归定义的卡特兰数
一个二叉树分为根节点,左子树,右子树
其中左子树和右子树也是二叉树,左右子树节点个数加起来等于\(n-1\)
设\(i\)个点能构成\(f_i\)个二叉树
我们枚举左子树有几个点可得
\(f_n=f_0*f_{n-1}+f_{1}*f_{n-2}+\ldots+f_{n-1}*f_{0}\)
这个和卡特兰数列的递归定义是一模一样的
凸多边形的三角划分
一个凸的\(n\)边形,用直线连接他的两个顶点使之分成多个三角形,每条直线不能相交,问一共有多少种划分方案
答案还是卡特兰数列
我们在凸多边形中随便挑两个顶点连一条边,这个凸多边形就会被分成两个小凸多边形,由于每条直线不能相交,接下来我们就只要求这两个小凸多边形的划分方案然后乘起来即可
和二叉树的构成问题一样,我们枚举大凸多边形被分成的两个小凸多边形的大小即可
阶梯的矩形划分
一个阶梯可以被若干个矩形拼出来
图示是两种划分方式


像下图是不合法的划分方式

问,一个\(n\)阶矩形有多少种矩形划分
答案仍然是卡特兰数列
我们考虑阶梯的每个角
如图

每个角一定是属于不同的矩形的,我们考虑和左下角属于一个矩形的是哪个角
这个矩形将这个梯形又分成两个小梯形,如图

于是我们又可以写出递推式了
和卡特兰数列的递归式是一样的
卡特兰数就将这么多吧
如有哪里讲得不是很明白或是有错误,欢迎指正
如您喜欢的话不妨点个赞收藏一下吧
卡特兰(Catalan)数入门详解的更多相关文章
- Linq之旅:Linq入门详解(Linq to Objects)
示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集 ...
- Quartz 入门详解
Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用.Quartz可以用来创建简单或为运行十个,百个, ...
- Redis快速入门详解
Redis入门详解 Redis简介 Redis安装 Redis配置 Redis数据类型 Redis功能 持久化 主从复制 事务支持 发布订阅 管道 虚拟内存 Redis性能 Redis部署 Redis ...
- Linq之旅:Linq入门详解(Linq to Objects)【转】
http://www.cnblogs.com/heyuquan/p/Linq-to-Objects.html Linq之旅:Linq入门详解(Linq to Objects) 示例代码下载:Linq之 ...
- Quartz 入门详解 专题
Cron-Expressions are used to configure instances of CronTrigger. Cron-Expressions are strings that a ...
- Linq之旅:Linq入门详解(Linq to Objects)(转)
http://www.cnblogs.com/heyuquan/p/Linq-to-Objects.html 示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细 ...
- SQL注入攻防入门详解
=============安全性篇目录============== 本文转载 毕业开始从事winfrm到今年转到 web ,在码农届已经足足混了快接近3年了,但是对安全方面的知识依旧薄弱,事实上是没机 ...
- SQL注入攻防入门详解(2)
SQL注入攻防入门详解 =============安全性篇目录============== 毕业开始从事winfrm到今年转到 web ,在码农届已经足足混了快接近3年了,但是对安全方面的知识依旧薄弱 ...
- [转]SQL注入攻防入门详解
原文地址:http://www.cnblogs.com/heyuquan/archive/2012/10/31/2748577.html =============安全性篇目录============ ...
随机推荐
- 转载:elastic5.x部署常见问题总结
原博文名称:ElasticSearch 5.0.0 安装部署常见错误或问题 原博文地址为:http://www.dajiangtai.com/community/18136.do?origin=csd ...
- Flume系列二之案例实战
Flume案例实战 写在前面 通过前面一篇文章http://blog.csdn.net/liuge36/article/details/78589505的介绍我们已经知道flume到底是什么?flum ...
- 每个人都要学的图片压缩终极奥义,有效解决 Android 程序 OOM
# 由来 在我们编写 Android 程序的时候,几乎永远逃避不了图片压缩的难题.除了应用图标之外,我们所要显示的图片基本上只有两个来源: 来自网络下载 本地相册中加载 不管是网上下载下来的也好,还是 ...
- 猿说python
一.简介 知识改变命运,程序改变世界.互联网时代潜移默化的改变着我们的生活,伴随技术的进步,我想下一个时代应该属于人工智能和机器学习,属于python. pytho ...
- 获取Android设备标识符
Android开发中有时候因业务需要客户端要产生一个唯一的标识符使服务器能识别某台Android设备,目前一般使用三种标识符分别为DeviceId.AndroidId.MAC地址. 获取DeviceI ...
- Ubuntu 搭建Zookeeper服务
1.下载安装包 官方下载地址http://apache.fayea.com/zookeeper/ 2.安装 安装前确保系统已安装过JDK,JDK安装过程可参照 2.1 解压下载好的tar.gz安装包到 ...
- 算法题解:最大或最小的K个数(海量数据Top K问题)
题目 输入 n 个整数,找出其中最小的 k 个数.例如输入4.5.1.6.2.7.3.8 这8个数字,则最小的4个数字是1.2.3.4. 初窥 这道题最简单的思路莫过于把输入的 n 个整数排序,排序之 ...
- 2019年最新超级有趣好玩的html+css网页布局课程,前端入门基础,html5+css3零基础入门课程-黑马程序员pink老师精心录制
大家好,我是黑马程序员pink老师!! 本次视频是前端零基础入门的课程,pink老师采取有趣好玩讲法,带你快乐的学习枯燥的html+css知识,学完之后让你能快速布局pc端页面.代码也可以讲的好玩有趣 ...
- tensorflow中添加L2正则化损失
方法有几种,总结一下方便后面使用. 1. tensorflow自动维护一个tf.GraphKeys.WEIGHTS集合,手动在集合里面添加(tf.add_to_collection())想要进行正则化 ...
- Dubbo学习系列之十五(Seata分布式事务方案TCC模式)
上篇的续集. 工具: Idea201902/JDK11/Gradle5.6.2/Mysql8.0.11/Lombok0.27/Postman7.5.0/SpringBoot2.1.9/Nacos1.1 ...