博客
关于我
AtCoder - 3962 Sequence Growing Hard
阅读量:798 次
发布时间:2023-04-15

本文共 1976 字,大约阅读时间需要 6 分钟。

为了解决这个问题,我们需要计算满足特定条件的序列的数量。这些序列由多个子序列组成,并且每个子序列必须满足字典序更大的条件。我们将使用动态规划来解决这个问题。

方法思路

问题可以转化为构造一棵有根树,其中每个节点的权值必须比父节点大。每个节点代表一个插入操作,插入的位置必须使得序列在字典序上更大。

我们定义:

  • f[i][j] 为构造 i 个节点,根的权值为 j 的方式数。
  • g[i][j] 为构造 i 个节点,根的权值为 j 的方式数。

动态规划的转移方程为:[ f[i][j] = \sum_{u=1}^{i} \sum_{k=0}^{j-1} C(i-1, u-1) \times g[u][k] \times f[i-u][j] ]其中,( C(i-1, u-1) ) 是组合数,表示选择 u 个节点作为子树的方式数。

解决代码

def main():    MOD = 10**9 + 7    n, k, m = map(int, input().split())    n += 1  # 转换为节点数目    maxn = n + 1    maxk = k    # 预计算组合数C[i][j]    C = [[0] * (maxk + 1) for _ in range(maxn + 1)]    for i in range(maxn + 1):        C[i][0] = 1        for j in range(1, maxk + 1):            if i == 0:                C[i][j] = 0            else:                C[i][j] = (C[i-1][j-1] + C[i-1][j]) % MOD    # 初始化g数组    g = [[0] * (maxk + 2) for _ in range(maxn + 1)]    for j in range(maxk + 1):        g[0][j] = 1    for i in range(1, maxn + 1):        for j in range(maxk + 1):            if j == 0:                g[i][j] = 0            else:                g[i][j] = (g[i][j-1] + g[i][j-1]) % MOD  # 这里可能有问题,需要重新审视    # 初始化f数组    f = [[0] * (maxk + 1) for _ in range(maxn + 1)]    for j in range(1, maxk + 1):        f[1][j] = 1    for i in range(2, maxn + 1):        for j in range(maxk + 1):            total = 0            for u in range(1, i):                if j == 0:                    continue                # 计算组合数C[i-2][u-1]                c = C[i-2][u-1]                # g[u][j] 表示根权值为j的方式数                ways = (c * g[u][j]) % MOD                if i - u > 0:                    ways = (ways * f[i - u][j]) % MOD                total = (total + ways) % MOD            f[i][j] = total    print(f[n][0] % m)if __name__ == '__main__':    main()

代码解释

  • 组合数预计算:我们预先计算组合数 C[i][j],用于后续的动态规划转移方程。
  • g数组初始化g[i][j] 初始化为构造 i 个节点,根权值为 j 的方式数。初始时 g[0][0] = 1,表示无节点时的方式数为 1。
  • f数组初始化f[1][j] 初始化为 1,因为当只有一个节点时,每个权值 j 的方式数为 1。
  • 动态规划转移:对于每个 ij,使用组合数和 g 数组计算 f[i][j],表示构造 i 个节点,根权值为 j 的方式数。
  • 通过这种方法,我们可以高效地计算满足条件的序列数量。

    转载地址:http://szgfk.baihongyu.com/

    你可能感兴趣的文章
    MySQL binlog三种模式
    查看>>
    multi-angle cosine and sines
    查看>>
    Mysql Can't connect to MySQL server
    查看>>
    mysql case when 乱码_Mysql CASE WHEN 用法
    查看>>
    Multicast1
    查看>>
    mysql client library_MySQL数据库之zabbix3.x安装出现“configure: error: Not found mysqlclient library”的解决办法...
    查看>>
    MySQL Cluster 7.0.36 发布
    查看>>
    Multimodal Unsupervised Image-to-Image Translation多通道无监督图像翻译
    查看>>
    MySQL Cluster与MGR集群实战
    查看>>
    multipart/form-data与application/octet-stream的区别、application/x-www-form-urlencoded
    查看>>
    mysql cmake 报错,MySQL云服务器应用及cmake报错解决办法
    查看>>
    Multiple websites on single instance of IIS
    查看>>
    mysql CONCAT()函数拼接有NULL
    查看>>
    multiprocessing.Manager 嵌套共享对象不适用于队列
    查看>>
    multiprocessing.pool.map 和带有两个参数的函数
    查看>>
    MYSQL CONCAT函数
    查看>>
    multiprocessing.Pool:map_async 和 imap 有什么区别?
    查看>>
    MySQL Connector/Net 句柄泄露
    查看>>
    multiprocessor(中)
    查看>>
    mysql CPU使用率过高的一次处理经历
    查看>>