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;
}
相关推荐
墨染天姬几秒前
[AI]OPENAI的PPO算法
人工智能·算法
万法若空4 分钟前
C++ <iomanip> 库全方位详解
开发语言·c++
c++之路5 分钟前
C++ 模板
linux·开发语言·c++
鸿儒51710 分钟前
记录一个C++ Windows程序移植到Linux系统的bug
开发语言·c++·bug
cici1587418 分钟前
含风光储燃的微电网能量管理系统(PSO优化)
算法
Das130 分钟前
图像色彩迁移技术算法及基本原理
算法
Titan202431 分钟前
C++11学习笔记
c++·笔记·学习
70asunflower34 分钟前
C/C++ 自定义函数的常用规范:从入门到工程实践
c语言·c++
发疯幼稚鬼34 分钟前
二叉树的广度优先遍历
c语言·数据结构·算法·宽度优先
谭欣辰36 分钟前
C++ DFS 与 BFS 剪枝方法详解
c++·算法·剪枝