判断有向图中是否有环-课程安排(Leetcode207)

207. Course Schedule

There are a total of n courses you have to take, labeled from 0 to n-1.

Some courses may have prerequisites, for example to take course 0 you have to first take course 1, which is expressed as a pair: [0,1]

Given the total number of courses and a list of prerequisite pairs, is it possible for you to finish all courses?
Example 1:

Input: 2, [[1,0]]
Output: true
Explanation: There are a total of 2 courses to take.
To take course 1 you should have finished course 0. So it is possible.
Example 2:

Input: 2, [[1,0],[0,1]]
Output: false
Explanation: There are a total of 2 courses to take.
To take course 1 you should have finished course 0, and to take course 0 you should
also have finished course 1. So it is impossible.
Note:

  1. The input prerequisites is a graph represented by a list of edges, not adjacency matrices. Read more about how a graph is represented.
  2. You may assume that there are no duplicate edges in the input prerequisites.

题解:

有 n 个课程(0~n-1),课程之前存在依赖关系,比如想要学习课程 0,就需要先学习课程 1;输入 n 个课程的依赖关系,判断是否可以将这 n 个课程全部完成。
例:
输入:2, [[1,0]];
2 表示两个课程, [[1,0]] 表示想要学习课程 1 需要先完成课程 0; 那我就先学 0 再学 1 就可以完成这 2 个课程了;
输出:true;
输入:2, [[1,0],[0,1]]
依然是两个课程,[[1,0],[0,1]] 表示想要学习课程 1 需要先完成课程 0;想要学习课程 0 需要先完成课程 1; 一脸懵逼发现死循环了,不可能将这两个课程全部完成;
输出:false;

分析:

这道题其实可以用有向图来实现;如图:

image.png

发现其实可以通过判断图中是否存在环来确认能否将课程全部完成;
啥?你咋发现的,这才俩点啊?好吧,那我们给四个课程关系,搞波大的:
image.png

好的,emmm这图挺大了(笑);分析下:
想要完成 0 :需要完成 1,3;
想要完成 1 :需要完成 2;
想要完成 2 :需要完成 3;
想要完成 3 :需要完成 1;
很明显,当我们要学习 1 的话,需要完成 2,学习 2 的话需要完成 3 ,学习 3 的话又要完成 1;这就死循环了,也就是图中的环;所以,只要构建的图中存在至少一个环的话,就一定无法完成所有课程;
那么重头戏来了,如何判断图中是否存在环呢?
我们用到了图的宽搜:https://www.jianshu.com/p/9828026169fc
对于图,我们可以删除入度为 0 的顶点,同时将从该点出发的相通的点入度减一;我们只要将图中所有入度为 0 的顶点全部删除,就可以判断图中是否有环了:如果图中所有顶点都能删除,说明所有的顶点最后入度都能减到 0 ,一定不存在环;反之,有环;而要实现这样一个过程,明显需要用到图的宽搜来解决;
image.png

遍历图中的顶点,对于每一个未被访问过的顶点进行宽搜:
宽搜的过程中,将入度为 1 的顶点存入队列中(因为入度为 1 说明该顶点只与当前队列头部的被宽搜的点相连,一旦删除队列头部的顶点,那么这个入度为 1 的点就是我们所需要的符合入度为 0 的点;);宽搜过程中,将该点相邻的顶点入度减一;当前队列头顶点宽搜结束后,将队列头删除,队列头变更为了新的入度为 0 的点,直到相邻点中不存在减一后入度为 0 的点,宽搜结束;
重复该过程,直到所有顶点均已被访问过;
这时,再来判断,所有的顶点的入度是否都为 0,都为 0 说明都被删除了,所以不存在环,反之,有环;

My Solution(C/C++)

#include <iostream>
#include <cstdio>
#include <vector>
#include <queue>

using namespace std;

struct GraphNode {
    int label;
    vector<GraphNode *> neighbors;
    GraphNode(int x) : label(x) {}
};

class Solution {
public:
    bool canFinish(int numCourses, vector<pair<int, int>> &prerequisites) {
        vector<GraphNode *> Graph;
        vector<int> visit;
        vector<int> degree;  //顶点的入度
        for (int i = 0; i < numCourses; i++) {
            Graph.push_back(new GraphNode(i));
            visit.push_back(0);
            degree.push_back(0);
        }
        for (int i = 0; i < prerequisites.size(); i++) {
            GraphNode *begin = Graph[prerequisites[i].first];
            GraphNode *end = Graph[prerequisites[i].second];
            begin->neighbors.push_back(end);
            degree[end->label] += 1;
        }
        for (int i = 0; i < numCourses; i++) {
            //printf("Graph: %d degree: %d\n", Graph[i]->label, degree[i]);
            if (visit[Graph[i]->label] == 0 && degree[Graph[i]->label] == 0) {
                BFS_Graph(Graph[i], visit, degree);
            }
        }
        for (int i = 0; i < numCourses; i++) {
            //printf("Graph: %d degree: %d\n", Graph[i]->label, degree[i]);
            if (degree[Graph[i]->label] != 0) {
                return false;
            }
        }
        return true;
    }
private:
    void BFS_Graph(GraphNode *node, vector<int> &visit, vector<int> &degree) {  //从当前顶点宽搜,判断是否存在环;
        queue<GraphNode *> q;
        //printf("Node: %d\n", node->label);
        q.push(node);
        visit[node->label] = 1;
        while (!q.empty()) {
            for (int i = 0; i < q.front()->neighbors.size(); i++) {
                //printf("Graph: %d degree: %d\n", q.front()->neighbors[i]->label, degree[q.front()->neighbors[i]->label]);
                if (degree[q.front()->neighbors[i]->label] == 1) {
                    q.push(q.front()->neighbors[i]);
                }
                visit[q.front()->label] = 1;
                degree[q.front()->neighbors[i]->label] -= 1;
            }
            q.pop();
        }
    }
};

int main() {
    int numCourses = 4;
    //int numCourses = 10;
    vector<pair<int, int>> prerequisites;
    prerequisites.push_back(make_pair(0, 1));
    prerequisites.push_back(make_pair(0, 3));
    prerequisites.push_back(make_pair(1, 2));
    prerequisites.push_back(make_pair(2, 3));
    prerequisites.push_back(make_pair(3, 1));
    //prerequisites.push_back(make_pair(5, 8));
    //prerequisites.push_back(make_pair(3, 5));
    //prerequisites.push_back(make_pair(1, 9));
    //prerequisites.push_back(make_pair(4, 5));
    //prerequisites.push_back(make_pair(0, 2));
    //prerequisites.push_back(make_pair(7, 8));
    //prerequisites.push_back(make_pair(4, 9));
    Solution s;
    printf("%d\n", s.canFinish(numCourses, prerequisites));
    system("pause");
    return 0;
}

结果

0
请按任意键继续. . .
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,504评论 4 365
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,898评论 1 300
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 110,218评论 0 248
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,322评论 0 214
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,693评论 3 290
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,812评论 1 223
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,010评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,747评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,476评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,700评论 2 251
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,190评论 1 262
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,541评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,206评论 3 240
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,129评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,903评论 0 199
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,894评论 2 283
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,748评论 2 274