博客
关于我
[bzoj5093][第二类斯特林数][NTT]图的价值
阅读量:86 次
发布时间:2019-02-26

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

为了解决这个问题,我们需要计算所有n个点的带标号简单无向图的价值之和。简单无向图的价值定义为每个点度数的k次方的和。给定n和k,我们需要对结果取模998244353。

方法思路

  • 问题分析

    • 简单无向图的价值是每个点度数的k次方的和。
    • 我们需要计算所有可能的简单无向图的总价值之和。
  • 关键思路

    • 每个点的度数可以独立计算,因此我们可以单独考虑每个点的贡献。
    • 总价值是所有点度数的k次方的和,可以转化为计算每个点在所有可能图中的度数的k次方的平均值,乘以n。
  • 数学推导

    • 使用斯特林数、组合数和生成函数来计算度数的k次方的和。
    • 斯特林数S(k, j)表示将k个元素分成j个非空子集的方式数。
    • 利用斯特林数和组合数来计算总和,并对结果取模。
  • 计算步骤

    • 预计算斯特林数S(k, j)。
    • 预计算阶乘j!。
    • 使用卢卡斯定理计算组合数C(n, j)。
    • 计算2的幂次。
  • 解决代码

    MOD = 998244353def main():    import sys    input = sys.stdin.read    data = input().split()    n = int(data[0])    k = int(data[1])    n -= 1  # 减1,因为每个点的度数最多为n-1        max_k = k    # 预计算斯特林数S(k, j)    st = [[0] * (max_k + 1) for _ in range(max_k + 1)]    st[0][0] = 1    for i in range(1, max_k + 1):        for j in range(1, i + 1):            st[i][j] = (st[i-1][j-1] + j * st[i-1][j]) % MOD        # 预计算阶乘    fact = [1] * (max_k + 1)    for i in range(1, max_k + 1):        fact[i] = fact[i-1] * i % MOD        # 计算斯特林数和各项的和    res = 0    for j in range(0, min(max_k, n) + 1):        s = st[k][j]        fj = fact[j]        # 计算C(n, j)        # 使用卢卡斯定理        def comb(n, k):            res = 1            while n > 0 or k > 0:                a = n % MOD                b = k % MOD                if b > a:                    return 0                # 计算C(a, b) mod MOD                numerator = 1                for i in range(b):                    numerator = numerator * (a - i) % MOD                denominator = 1                for i in range(b):                    denominator = denominator * (i + 1) % MOD                inv_denominator = pow(denominator, MOD-2, MOD)                c = numerator * inv_denominator % MOD                res = res * c % MOD                n = n // MOD                k = k // MOD            return res                c_nj = comb(n, j)        if c_nj == 0:            continue        # 计算2^{n - j}        power = pow(2, n - j, MOD)        term = s * fj % MOD        term = term * c_nj % MOD        term = term * power % MOD        res = (res + term) % MOD        print(res % MOD)if __name__ == '__main__':    main()

    代码解释

  • 预计算斯特林数

    • 使用动态规划计算斯特林数S(k, j),其中0 ≤ j ≤ k。
  • 预计算阶乘

    • 计算阶乘j!,用于后续计算斯特林数和组合数。
  • 组合数计算

    • 使用卢卡斯定理计算大数下的组合数C(n, j),适用于n很大的情况。
  • 计算2的幂次

    • 使用快速幂计算2^{n-j},处理大指数取模。
  • 求和

    • 对每个j,计算对应的斯特林数、阶乘、组合数和2的幂次的乘积,累加到结果中。
  • 通过以上步骤,我们可以高效地计算所有n个点带标号简单无向图的价值之和,并对结果取模998244353。

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

    你可能感兴趣的文章
    Pandas:对给定列求和 DataFrame 行
    查看>>
    Pandas、Matplotlib、Pyecharts数据分析实践
    查看>>
    Pandas中文官档~基础用法2
    查看>>
    Pandas中文官档~基础用法6
    查看>>
    pandas交换两列
    查看>>
    pandas实战:电商平台用户分析
    查看>>
    Pandas库常用方法、函数集合
    查看>>
    pandas打乱数据的顺序
    查看>>
    pandas改变一列值(通过apply)
    查看>>
    Pandas数据分析的环境准备
    查看>>
    Pandas数据可视化怎么做?用实战案例告诉你!
    查看>>
    Pandas数据处理与分析教程:从基础到实战
    查看>>
    Pandas数据结构之DataFrame常见操作
    查看>>
    pandas整合多份csv文件
    查看>>
    pandas某一列转数组list
    查看>>
    Pandas模块,我觉得掌握这些就够用了!
    查看>>
    Pandas玩转文本处理!
    查看>>
    SpringBoot 整合 Mybatis Plus 实现基本CRUD功能
    查看>>
    pandas的to_sql方法中使用if_exists=‘replace‘
    查看>>
    pandas读取parquet报错
    查看>>