zoukankan      html  css  js  c++  java
  • FJOI 2020 Day1 T3 异构序列码性态问题

    题意简述

    有两个序列 (a, b) 以及两个辅助栈 (s_1, s_2)

    初始时 (a) 的元素依次为 ([1, 2, 3, ldots , n]),而 (s_1, s_2, b) 均为空。

    每次可以执行如下三种操作之一:

    1. (a) 非空,将 (a) 的最后一个元素删除,插入到第一个栈 (s_1) 的栈顶。
    2. (s_1) 非空,将 (s_1) 的栈顶弹出,插入到第二个栈 (s_2) 的栈顶。
    3. (s_2) 非空,将 (s_2) 的栈顶弹出,插入到序列 (b) 的第一个元素之前。

    需要特别注意的是,任意时刻必须保证 (s_2) 从栈顶到栈底的序列是递减的。

    当所有元素均被插入到序列 (b) 中,停止操作,我们称得到了序列 (b)

    请问在所有的 (n!)(1 sim n) 的排列中,有哪些是不能在这个过程中得到的。

    样例:当 (n le 3) 时答案为 (0),当 (n = 4) 时答案为 (2)

    题解

    考虑统计能够得到的序列 (b) 的个数,最后用 (n!) 减去。

    分析序列 (b) 需要满足的性质,通过手动枚举发现当 (n = 4) 时仅有这两个序列不能被得到:

    1. 序列 ([3, 1, 4, 2])
    2. 序列 ([3, 2, 4, 1])

    同时我们可以写出一个,给出最终序列 (b),判断它是否能被得到的程序,每次贪心地用最少的步数把目标元素移到 (b) 的开始即可。

    利用这个程序,枚举所有的全排列,我们容易得到 (n = 5) 或更大时,无法被得到的序列。

    稍加观察能够发现,它们也都满足性质:存在不一定连续的子序列,满足其中的元素离散化后依次等于 ([3, 1, 4, 2])([3, 2, 4, 1])

    为了更准确地验证,可以写一个 (mathcal O (n^4)) 枚举子序列并判断的程序,枚举全排列就能验证这个性质。

    所以得到唯一的性质:(b) 中不存在任何子序列满足其中的元素离散化后依次等于 ([3, 1, 4, 2])([3, 2, 4, 1])

    我们考虑按照值从小到大加入元素,一开始为空序列,然后依次加入 (1, 2, 3, ldots)

    观察到在此时前面的条件就等价于,在某个位置插入当前的数后,之后就不能在这个位置后面的元素中的空隙中插入更大的元素了。

    这是因为当前时刻前插入的数都比当前数小,对应 (1, 2),而当前数对应 (3),当前时刻后插入的数对应 (4),不能在 (1, 2) 之间插入。

    举个例子:

    1. (left[ oxed{1} ight])
    2. (left[ oxed{2}, oxed{1} ight])
    3. (left[ oxed{3}, oxed{2, 1} ight])
    4. (left[ oxed{3}, oxed{2, 1}, oxed{4} ight])
    5. (left[ oxed{3}, oxed{5}, oxed{2, 1, 4} ight])
    6. (left[ oxed{3}, oxed{5}, oxed{2, 1, 4}, oxed{6} ight])
    7. (left[ oxed{7}, oxed{3, 5, 2, 1, 4, 6} ight])

    也就是,如果已经确定一段元素中间不能插入其它元素了,那么就把它们用方框框住。

    这导出一个 DP 方程:令 (mathrm{f}[i][j]) 表示目前加入了 (1 sim i),并且当前有 (j) 个方框时的方案数。

    (n ge 2) 时,有边界条件 (mathrm{f}[2][2] = 2) 以及转移方程:

    [egin{aligned} mathrm{f}[i][j] & o_+ mathrm{f}[i + 1][k] : mathrm{Coef} \ mathrm{Coef} &= egin{cases} 1 & , 2 le k le j \ 2 & , k = j + 1 \ 0 & , ext{otherwise} end{cases} end{aligned} ]

    暴力转移得到 (mathcal O (n^3)) 的做法。

    还可以得到 (mathrm{f}[i][j] = mathrm{f}[i][j + 1] + 2 mathrm{f}[i - 1][j - 1] - mathrm{f}[i - 1][j]),得到 (mathcal O (n^2)) 的做法,这个递推式将会是接下来的关键。

    观察上图,可以发现就是递推式的图形化体现,(mathrm{f}[i][j]) 的值为上图中从 ((2, 2)) 走到 ((i, j)) 的带权路径数量乘以 (2)

    要求的即是对于所有 (2 le i le n),从 ((2, 2)) 走到 ((n, i)) 的带权路径数量之和乘以 (2)

    一个观察:对于一行的从 ((2, 2)) 走到 ((n, i)) 的带权路径数量之和,等于从 ((2, 2)) 走到 ((n + 1, 2)) 的带权路径数量。

    所以要求从 ((2, 2)) 走到 ((n + 1, 2)) 的带权路径数量,乘以 (2) 就是能被得到的序列 (b) 的数量。

    令上图中的下标都偏移 (-2),即从 ((0, 0)) 开始,到达 ((n - 1, 0)) 的方案数。

    (n' = n - 1),所求即是 (displaystyle sum_{i = 0}^{n'} 2^i {(-1)}^{n' - i} inom{n' + i}{n' - i} C_i),其中 (C_i) 为卡特兰数,(C_1 = 1, C_2 = 2, C_3 = 5)

    意义为枚举往右下走了 (i) 步,则往下是 ((n' - i)) 步,贡献 (2^i {(-1)}^{n' - i})

    不同的走法有 (C_i) 种,插入向下走的 (n' - i) 步有 (displaystyle inom{n' + i}{n' - i}) 种方法。

  • 相关阅读:
    PowerDesigner16 破解
    双向链表的实现与操作(C语言实现)
    【HDOJ 2063】过山车
    gcc 源代码分析-前端篇2
    Android4.0.4-在build.prop中添加属性的方法【转】
    安卓系统手动添加虚拟按键教程【转】
    android 添加新的键值,自定义按键-2【转】
    android 添加新的键值,自定义按键【转】
    Android下添加新的自定义键值和按键处理流程【转】
    android监听虚拟按键的显示与隐藏【转】
  • 原文地址:https://www.cnblogs.com/PinkRabbit/p/FJOI2020D1T3.html
Copyright © 2011-2022 走看看