【洛谷P1754 球迷购票问题】题解


卡特兰数经典 AB 分拆问题。

分析:

题意相当于排列 nAnB,使得相邻 AB(有序!)消掉,然后左右元素并到一起再消,最后消完的序列个数。

AB 为一个组“1”,AB 自嵌套一次为一个组“2”(即 AABB),以此类推。

后面大多数数字指组“数字”

题意即转换为一个数 n,求 n 分解成若干个正整数之和的方案数。

神犇到这一步就可以切掉了吧。

我们这里考虑隔板法:

两个 1 当然可以合并 2=1+1),ab 当然可以合并 a+b,问题转换为有 n1 有多少种合并方案。

n 个数的方案数为 f(n)

考虑将 f(n) 分解为 f(x0+y0)

使用隔板:

  • 当隔板在最左侧时,x0=0f(0)=1y0=n,因为要求合并,所以有 n1 种,由乘法原理知,此步答案为 f(0)f(n1)
  • 隔板向右移动一格,x0=1,也就是 f(1)y0=n1,同理是 n2 种,由乘法原理知,此步答案为 f(1)f(n2)
  • 归纳一下,第 i 步为 f(i)f(ni1)
  • 最后一步显然是 f(n1)f(0);左右对称。

于是得出递推式:

f(n)=f(0)f(n1)+f(1)f(n2)+f(2)f(n3)++f(i)f(ni1)++f(n1)f(0)

朴素 dp 即可:

#include<iostream>
using namespace std;
const int N=25;
typedef long long ll;
ll n,dp[N];
int main()
{
	cin>>n;
	dp[0]=1;
	for (int i=1;i<=n;i++)
		for (int j=1;j<=n;j++)
			dp[i]+=dp[j-1]*dp[i-j];
	cout<<dp[n];
	return 0;
}

但是再深入一步,会发现 f(n)=f(0)f(n1)+f(1)f(n2)+f(2)f(n3)++f(i)f(ni1)++f(n1)f(0) 的这个 f(n) 正好就是卡特兰数 Cn,这个公式正好是一个卡特兰数的递推式。

posted @   yspm  阅读(171)  评论(1编辑  收藏  举报
编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】
😅​
点击右上角即可分享
微信分享提示