P4017 最大食物链计数

题目:P4017 最大食物链计数

知识点:链式前向星建图、同余原理、拓扑排序、动态规划

CODE:

cpp 复制代码
#include <bits/stdc++.h>
#define MOD 80112002
using namespace std;

static const int N = 5005;
static const int M = 500005;

vector<int> head(N, 0);
vector<int> nxt(M, 0);
vector<int> to(M, 0);

vector<int> idgr(N, 0);
vector<int> odgr(N, 0);

vector<int> dp(N, 0);


int main() {
    int n, m;
    cin >> n >> m;
    int cnt = 1;
    while (m--) {
        int f, t;
        cin >> f >> t;
        // idgr & odgr
        odgr[f]++;
        idgr[t]++;
        // f -> t
        to[cnt] = t;
        nxt[cnt] = head[f];
        head[f] = cnt++;
    }

    // ...
    queue<int> q;
    int ans = 0;
    for (int i = 1; i <= n; i++) {
        if (idgr[i] == 0) {
            q.emplace(i);
            dp[i] = 1;
            if (odgr[i] == 0) {
                ans++;
            }
        }
    }

    while (!q.empty()) {
        int f = q.front();
        q.pop();
        int e = head[f];
        while (e != 0) {
            // 边 e
            int v = to[e];
            idgr[v]--;
            // dp[v] += dp[f];
            dp[v] = ((dp[v] % MOD) + dp[f]) % MOD;
            if (idgr[v] == 0) {
                q.emplace(v);
                if (odgr[v] == 0) {
                    ans = ((ans % MOD) + dp[v]) % MOD;
                }
            }
            e = nxt[e];
        }
    }

    cout << ans << endl;

}

易错点:

①对于孤立点,要计算一次,42line
②对于拓扑排序的终点,是当入度变为 0 时才去判断是否为终点,而不是只要出度为 0 就记录,如下为错误示例,这会导致入度为复数的终点重复计算
cpp 复制代码
// ...
if (idgr[v] == 0) {
  q.emplace(v);
}
if (odgr[v] == 0 {
  ans = ((ans % MOD) + dp[v]) % MOD;
}
相关推荐
程序员小明儿2 小时前
量子计算探秘:从零开始的量子编程与算法之旅 · 第四篇
算法·量子计算
程序员小明儿2 小时前
量子计算探秘:从零开始的量子编程与算法之旅 · 第一篇
算法·量子计算
不想看见4042 小时前
Add Strings数字处理--力扣101算法题解笔记
数据结构·算法·leetcode
sycmancia2 小时前
C++——对象模型分析
开发语言·c++
云泽8082 小时前
C++ STL set 容器全解析:从基础用法、算法实践到云同步实战
开发语言·c++·算法
山上三树2 小时前
C++ 智能指针详解与代码示例
开发语言·c++
absunique3 小时前
复杂度分析中的常数优化与性能重构的技术6
算法·重构
载数而行5203 小时前
QT系列,对象树 栈和堆 QDebug以及日志打印
c++·qt·学习
初次攀爬者3 小时前
力扣解题-74. 搜索二维矩阵
算法·leetcode