卡特兰数(catalan)总结
卡特兰数的公式
递推公式1:$f(n)=\sum \limits_{i=0}^{n-1}f(i)*f(n-i-1)$
递推公式2:$f(n)=\frac{f(n-1)*(4*n-2)}{n+1}$
组合公式1:$f(n)=\frac{C_{2n}^{n}}{n+1}$
组合公式2:$f(n)=C_{2n}^{n}-C_{2n}^{n-1}$
关于卡特兰数的题目
1. 有限制的网格方案数 eg网格
利用组合数的思想:
对于长N宽M的网格(下图2),方案数为 $C_{n+m}^{m}-C_{n+m}^{m-1}$
理解:走到(n,m)这个点总共要走n+m步,其中有m步一定是向上的,所以$C_{n+m}^{m}$这是所有情况
但有不合法的情况,且不合法的一定经过绿线,将原图形沿其翻折,相当于走到c点,此时总n+m步不变,但只有m-1步是向右的
所以$C_{n+m}^{m-1}$是不合法的
(借用kaola学长的图)
对于N×N的网格就是卡特兰数了,如图一
本题先将式子化简,然后将其分解质因数,消去除法,最后乘上每个质数的个数次方就好
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
int n,m,num,p[],v[];
int sum[];
void prime(int x)
{
for(int i=;i<=x;i++)
{
if(!v[i]) {v[i]=i;p[++num]=i;}
for(int j=;j<=num;j++){
if(p[j]>v[i]||i*p[j]>x) continue;
v[i*p[j]]=p[j];
}
}
}
int len=,ans[];
void mul(int x)
{
int k=;
for(int i=;i<=len;i++)
{
ans[i]=ans[i]*x+k;
k=ans[i]/;
ans[i]%=;
if(k>&&i==len) len++;
}
}
int main()
{
ans[]=;
scanf("%d%d",&n,&m);
prime(n+m+);
int t=n+-m;
while(t>)
{
sum[v[t]]++;
t/=v[t];
}
for(int i=n+m;i>=n+;i--)
{
t=i;
while(t>)
{
sum[v[t]]++;
t/=v[t];
}
}
for(int i=;i<=m;i++)
{
t=i;
while(t>)
{
sum[v[t]]--;
t/=v[t];
}
}
for(int i=;i<=num;i++)
for(int j=;j<=sum[p[i]];j++)
mul(p[i]);
for(int i=len;i>=;i--)
printf("%d",ans[i]);
puts("");
}
2.有趣的数列
其实这个也可以理解为上一个网格,将偶数位记为向右走一步,奇数位记为向上走一步,,偶数位之和大于奇数位之和,就是不能越过绿线
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#define ll long long
using namespace std;
const int maxn=;
int n,mod,num;
ll p[maxn];int v[maxn];
ll sum[maxn];
void prime(int x)
{
for(int i=;i<=x;i++)
{
if(!v[i]) {v[i]=i;p[++num]=i;}
for(int j=;j<=num;j++){
if(p[j]>v[i]||i*p[j]>x) break;
v[i*p[j]]=p[j];
}
}
}
ll qpow(int a,int b)
{
ll ans=;
while(b)
{
if(b&) ans=ans*a%mod;
a=a*a%mod;
b>>=;
}
return ans%mod;
}
int main()
{
scanf("%d%d",&n,&mod);
prime(*n+);
for(int i=*n;i>=n+;i--)
{
int t=i;
while(t>)
{
sum[v[t]]++;
t/=v[t];
}
}
for(int i=;i<=n;i++)
{
int t=i;
while(t>)
{
sum[v[t]]--;
t/=v[t];
}
}
ll ans=;
for(int i=;i<=num;i++)
if(sum[p[i]])
ans=ans*qpow(p[i],sum[p[i]])%mod;
printf("%lld\n",ans);
}
3.树屋阶梯

我们不妨手模样例,若扣去左下角直角所在矩形,
图一和图四的方案数为右面的2块的方案数×上面的0块的方案数,即为$f(3)+=f(2)*f(0)$
同理图二和图五为$f(3)+=f(0)*f(2)$ 图三为$f(3)+=f(1)*f(1)$
由此可得 $f(n)=\sum \limits_{i=0}^{n-1}f(i)*f(n-i-1)$ 卡特兰数公式1
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
int n,num,p[],v[];
int sum[];
void prime(int x)
{
for(int i=;i<=x;i++)
{
if(!v[i]) {v[i]=i;p[++num]=i;}
for(int j=;j<=num;j++){
if(p[j]>v[i]||i*p[j]>x) break;
v[i*p[j]]=p[j];
}
}
}
int len=,ans[];
void mul(int x)
{
int k=;
for(int i=;i<=len;i++)
{
ans[i]=ans[i]*x+k;
k=ans[i]/;
ans[i]%=;
if(k>&&i==len) len++;
}
}
int main()
{
ans[]=;
scanf("%d",&n);
prime(*n+);
for(int i=*n;i>=n+;i--)
{
int t=i;
while(t>)
{
sum[v[t]]++;
t/=v[t];
}
}
for(int i=;i<=n;i++)
{
int t=i;
while(t>)
{
sum[v[t]]--;
t/=v[t];
}
}
for(int i=;i<=num;i++)
for(int j=;j<=sum[p[i]];j++)
mul(p[i]);
for(int i=len;i>=;i--)
printf("%d",ans[i]);
puts("");
}
关于卡特兰数的其他应用
1.出栈入栈问题:1,2,~n个数经过一个栈,合法的出栈序列$Cat(n)$
(引用学长的课件)出栈次序是卡特兰数的一个应用。 我们将入栈视为+1,出栈视为-1,则限制条件为在任意位置前缀和不小于0 。 我们讨论这个问题与卡特兰数有什么关系。 为了方便,我们按入栈的先后顺序将各个元素由1到n编号。 假设最后一个出栈的数为k。 则在k入栈之前,比k小的数一定全部出栈,所以这部分方案数为h(k-1)。 在k入栈之后,比k大的数在k入栈之后入栈,在k出栈之前出栈,所以这部分的方案数为h(n-k)。 这两部分互不干扰,则方案数为h(k-1)*h(n-k) 枚举k,得到的公式就是卡特兰数的递推公式。
2.左括号与右括号的匹配问题:n个左括号和n个右括号组成的合法括号序列$Cat(n)$
跟入栈出栈的理解是一样的
3.n个节点构成的二叉树的方案数为$Cat(n)$
假设左子树有$i$个节点,右子树有$n-i-1$个节点,i从0到n-1,根据乘法原理
可得公式1$f(n)=\sum \limits_{i=0}^{n-1}f(i)*f(n-i-1)$
卡特兰数(catalan)总结的更多相关文章
- 卡特兰数 Catalan数 ( ACM 数论 组合 )
卡特兰数 Catalan数 ( ACM 数论 组合 ) Posted on 2010-08-07 21:51 MiYu 阅读(13170) 评论(1) 编辑 收藏 引用 所属分类: ACM ( 数论 ...
- 浅谈卡特兰数(Catalan number)的原理和相关应用
一.卡特兰数(Catalan number) 1.定义 组合数学中一个常出现在各种计数问题中出现的数列(用c表示).以比利时的数学家欧仁·查理·卡特兰的名字来命名: 2.计算公式 (1)递推公式 c[ ...
- 卡特兰数 catalan number
作者:阿凡卢 出处:http://www.cnblogs.com/luxiaoxun/ 本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留 ...
- 卡特兰数(Catalan Number) 算法、数论 组合~
Catalan number,卡特兰数又称卡塔兰数,是组合数学中一个常出现在各种计数问题中出现的数列.以比利时的数学家欧仁·查理·卡塔兰 (1814–1894)命名. 卡特兰数的前几个数 前20项为( ...
- 卡特兰数 Catalan 笔记
一.公式 卡特兰数一般公式 令h(0)=1,h(1)=1,catalan数满足递推式.h(n) = h(0)*h(n-1)+h(1)*h(n-2) + ... + h(n-1)h(0) (n>= ...
- 卡特兰数(Catalan)及其应用
卡特兰数 大佬博客https://blog.csdn.net/doc_sgl/article/details/8880468 卡特兰数是组合数学中一个常出现在各种计数问题中出现的数列. 卡特兰数前几项 ...
- ACM数论-卡特兰数Catalan
Catalan 原理: 令h(0)=1,h(1)=1,catalan 数满足递归式: (其中n>=2) 另类递推公式: 该递推关系的解为: (n=1,2,3,...) 卡特兰数的应用实质上都是递 ...
- 卡特兰数(Catalan Number) 学习笔记
一.三个简单的问题 1.给定一串长为2n的01序列,其中0和1的数量相等,满足任意前缀中0的个数不少于1的个数,求序列的个数 2.给出一串长为n的序列,按顺序将他们进栈,随意出栈,求最后进出栈的方案 ...
- 【知识总结】卡特兰数 (Catalan Number) 公式的推导
卡特兰数的英文维基讲得非常全面,强烈建议阅读! Catalan number - Wikipedia (本文中图片也来源于这个页面) 由于本人太菜,这里只选取其中两个公式进行总结. (似乎就是这两个比 ...
- 【2020.12.01提高组模拟】卡特兰数(catalan)
题目 题目描述 今天,接触信息学不久的小\(A\)刚刚学习了卡特兰数. 卡特兰数的一个经典定义是,将\(n\)个数依次入栈,合法的出栈序列个数. 小\(A\)觉得这样的情况太平凡了.于是,他给出了\( ...
随机推荐
- Genymotion 的那些事
一.什么是Genymotion? Genymotion是一套完整的工具,它提供了Android虚拟环境. 支持Windows.Linux和Mac OS等操作系统. 在Android开发中可以代替安卓模 ...
- 隐藏/显示jeecg-boot 后端管理页面的右侧的系统设置
登录后台,通过添加一个下拉选项[系统设置]来控制系统的后侧系统设置,布局如下: 修改UserMenu.vue文件 1.全局搜索“账户设置”,找到对应的vue文件:UserMenu.vue 2.添加[系 ...
- rpm包安装过程中依赖问题“libc.so.6 is needed by XXX”解决方法-转
原文:http://raksmart.idcspy.com/781 在CentOS上的Canon LBP2900安装打印机驱动,中间遇到了一些问题,主要是安装rpm包出现的依赖问题,现在解决了,现在简 ...
- [COCI2019] Mobitel
题目 显然不小于\(n\)这个东西我们不是很好搞,考虑正难则反,求出有多少条路径小于\(n\),之后拿\(C_{n+m}^m\)一减就好了 于是状态为\(dp[i][j][k]\)表示到\((i,j) ...
- appscan如何扫描移动应用APP
1.前置条件:让手机和电脑处于同一WIFI下 1打开appscan,选择手动探索/外部设备. 2在弹出的对话框页面点击右上角“记录代理配置”. 3在弹出的页面选择记录代理页签,设置Appscan代理端 ...
- nosql BASE
- Windows API 第 10篇 SearchTreeForFile
函数原型:BOOL SearchTreeForFile( PSTR RootPath, //系统查找的起始路径, PSTR InputPathName, ...
- [转]浅谈C#中常见的委托
一提到委托,浮现在我们脑海中的大概是听的最多的就是类似C++的函数指针吧,呵呵,至少我的第一个反应是这样的. 关于委托的定义和使用,已经有诸多的人讲解过,并且讲解细致入微,尤其是张子阳的那一篇.我就不 ...
- ODOO 新API修饰符
Odoo8中,API接口分为traditaional style和record style两种类型: traditional style指的就是我们在7.0中使用的类型,def(self,cr,uid ...
- Schedule(Hackerrank Quora Haqathon)
题目链接 Problem Statement At Quora, we run all our unit tests across many machines in a test cluster on ...