题解 洛谷 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 了

代码

cpp 复制代码
#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;
}
相关推荐
xwl121221 小时前
10.6 作业
数据结构·算法
青草地溪水旁21 小时前
VSCode C/C++ 构建任务配置文件 `tasks.json` 全字段深度解析
c语言·c++·vscode
爱和冰阔落1 天前
C++模板进阶 非类型模板参数 模板的特化 分离编译的深入探索
c++·面试·编译原理·模板
胡小禾1 天前
JDK17和JDK8的 G1
jvm·算法
胖咕噜的稞达鸭1 天前
算法入门:专题攻克一---双指针(3)有效三角形的个数 查找总价格为目标值的两个商品(剑指offer题目)
算法
charlie1145141911 天前
精读C++20设计模式:行为型设计模式:中介者模式
c++·学习·设计模式·c++20·中介者模式
楼田莉子1 天前
Qt开发学习——QtCreator深度介绍/程序运行/开发规范/对象树
开发语言·前端·c++·qt·学习
逻辑留白陈1 天前
Adaboost进阶:与主流集成算法对比+工业级案例+未来方向
算法
Learn Beyond Limits1 天前
Mean Normalization|均值归一化
人工智能·神经网络·算法·机器学习·均值算法·ai·吴恩达
oioihoii1 天前
超越 std::unique_ptr:探讨自定义删除器的真正力量
c++