Bootstrap

题解 洛谷 Luogu P1113 杂务 图论 BFS C++

题目

传送门

P1113 杂务 - 洛谷 | 计算机科学教育新生态https://www.luogu.com.cn/problem/P1113

思路

从工作之间的指向性关系不难看出这是基于有向图的问题

边从前置工作通向后置工作,表示完成了前置任务后才能完成后置工作

再模拟这个过程,可以发现这个过程非常像搜索,且更像 BFS

前置工作少的大致在更先搜到的层,前置工作多的大致在更后搜到的层

设 dist[i] 为从开始搜索一直到完成工作 i 所需的时间

t[i] 为完成工作 i 本身需要的时间,若 x 通向 y,则 dist[y] = dist[x] + t[y]

dist[i] 的最大值就是所求答案

但这和一般的 BFS 又不一样

以输入样例为例,2 为队头时,不能直接入队 2 通向的 5,因为 4 也通向 5

入队条件

那我们不禁要想,什么时候才能入队 5 呢?

必须完成所有前置工作才能入队 5

即当队头 x 通向 5 且 x 是 5 的最后一个前置工作时,才能入队 5

维护前置工作信息

我们又怎么知道 x 是不是 5 的最后一个前置工作呢?

额外维护一个数组 from[i],表示通向 i 的节点,也就是 i 的前置工作

对于输入样例,from[5] = { 2,4 },队头为 2 时从 from[5] 中删掉 2

队头为 4 时发现 4 是 5 的最后一个前置工作了,那么此时就可以入队 5 了

Dist[i] 计算

dist[5] 又怎么计算呢?

显然是所有通向 5 的节点的 dist 值的最大值再加上 t[5],即 dist[2]、dist[4] 的最大值再加上 t[5]

搞清楚以上问题后,就可以愉快地写 BFS 了

代码

#include <algorithm>
#include <vector>
#include <iostream>
using namespace std;
const int N = 10005;
vector<int> from[N], to[N]; //from 存前置工作,to 存后置工作
int n, t[N], dist[N], q[N], head, tail = -1, ans;
void BFS()
{
    while (head <= tail)
    {
        int x = q[head++];
        for (auto X : to[x]) //小 x 是出发点 (前置工作),大 X 是到达点 (后置工作)
        {
            dist[X] = max(dist[X], dist[x] + t[X]); //dist[X] 是 dist[x] 的最大值加上 t[X]
            ans = max(ans, dist[X]);
            if (from[X].size() == 1) q[++tail] = X; //x 是 X 的最后一个前置工作,那么可以入队 X
            from[X].erase(find(from[X].begin(), from[X].end(), x)); //完成了前置工作 x,将 x 从 X 的前置任务列表中删掉
        }
    }
}
int main()
{
    scanf("%d", &n);
    for (int i = 1; i <= n; i++)
    {
        scanf("%d%d", &t[i], &t[i]);
        int From;
        while (scanf("%d", &From) && From)
            from[i].push_back(From), to[From].push_back(i); //建图
        if (from[i].empty()) //没有前置工作的工作,是搜索的起点,入队。且其 dist 值要更新成完成该工作本身需要的时间
            q[++tail] = i, dist[i] = t[i];
    }
    BFS();
    printf("%d", ans);
    return 0;
}
;