zoukankan      html  css  js  c++  java
  • 《卡特兰数》

    模型一:进出序列。

    求解:给定n个的数,有多少种出栈序列?(抽象成一个有n个1和n个-1组成的字串,且前k个数的和均不小于0,那这种字串的总数为多少?)

    容斥可得 ans(合法) = ans(all) - ans(不合法)
    ans(all) = C(2n,n) //任选n个位置放1.

    首选,我们将这个字符串看成二进制,那么对于每个不合法的方案数,一定是前缀和 = -1的第一个位置,也就是2 * m + 1的位置处是-1.

    前面m个1,m个-1.那么剩下(2 * n - (2 * m + 1)) = 2 * (n - m) - 1个位。

    我们将这些位都翻转,可得这个字符串由n + 1个 - 1和n - 1个1组成,并且我们可以发现,对于由(n + 1)个-1和(n - 1)个+1组成的字符串,一定都存在前缀和 -1的第一个位置,我们将剩下的位置反转。

    就都对应着n 个 1和n 个 -1组成的不合法字符串,所以对于所有(n + 1)个-1和(n - 1)个1组成的字符串,都对应着n 个 1和n 个 -1组成的不合法字符串.

    所以ans(不合法) = C(2n,n + 1)//任选n + 1个位置放-1.

    所以ans(all) = C(2n,n) - C(2n,n + 1) = C(2n,n) / (n + 1).

     扩展:一个有个 1 和 m 个 -1(n >= m) 组成的字串,且前k个数的和均不小于0,那这种字串的总数为多少?

    这里其实是类似的,但是我们如果还去翻转后缀,那么就很难判断增减的数量。

    所以我们考虑去翻转前缀,对于任何一个 < 0的第一个前缀,翻转后变成n + 1个1,m - 1个-1.

    并且翻转之后保证n + 1 > m - 1,因为n + 1个1和m - 1个-1组成的序列肯定存在前缀和 > 0的情况,也就对应着原来的每种不合法的状态。

    所以ans = C(n + m,m) - C(n + m,n + 1) * (n! * m!)如果每个1,-1之间不同还需要乘上这个阶乘的方案数,内部的排序。

  • 相关阅读:
    转载完美解决国内访问GitHub速度太慢的难题
    leetcodedp最长子序列问题isSubsequence
    三级域名
    2021年终总结
    leetcodedp背包子集问题
    dp01背包
    springboot整合mybatis(使用的baomidou的mybatisplusbootstarter)实现多数据源切换
    leetcode完全背包518.零钱兑换II
    mybatislogplugin
    Mybatis log plugin插件破解修复版 MyBatis Log Plugin License Authorization Failed
  • 原文地址:https://www.cnblogs.com/zwjzwj/p/15037961.html
Copyright © 2011-2022 走看看