1选课问题

  Leetcode上有这样一道题:有代号0,1,2……n-1的n门课程。其中选择某些课程需要另一些课程作为前提条件。用一组pair来表示这些条件:[1,0],[1,2],表示如果要选修课程1,必须先选修课程0和课程2。问是否可能把这n门课程都选修一遍。

  这个问题看起来相当复杂。难点在于,一门课程可能需要多门课程作为前提条件,这样就很难找到一条可以不重复地遍历所有课程的方法。

  比较笨的方法就是先找出那些不需要前提条件的课程,然后由他们出发将那些只需要他们作为前提条件的课程修完。再循环往复。一直到某一次遍历之后,并没有修到新的课程。最后再检查是否还有课程没能修够。这样的话,最坏情况,我们是时间复杂度最坏能达到O(N^3)。使用了unordered_multimap作为辅助工具,代码被Accept了。但是运行时间是最好的coder的10倍。于是,细细研究他人的算法。

2 BFS

  其实。这道题难在用一维数据结构来表示这些前提条件,很容易增大时间复杂度。于是,考虑用二维数据结构来表示。

  图!而题中给的条件明显是一对多的。那么,适合用邻接链表法表示。采用数据结构vector<unordered_set<int>> map来表示。vector的每个元素表示由某门课程作为前提条件的一组课程。每门课程需要的前提条件可以用一组向量表示vector<int> de。我们用入度称呼这个值,也就是说在图中,进入该节点有多少条路径。那么对于那些不需要前提条件就完成的课程,其入度为0。

  BFS法,广度优先。我们先找到一门入度为0的课,然后由它作为前提条件的课程的入度都可以减1,因为这门课程学到了,就等于不需要这个前提条件了。而以它为前提的课程就是该节点的邻接节点。用这种方法,我们每次学到一门课程,N次就可以学完所有的课程,如果某一次没有学到新的课程,那么最终肯定是无法学完的。

  上代码:    

class Solution {
public:
bool canFinish(int n, vector<pair<int, int>>& pre) {
vector<unordered_set<int> > gra(n);
vector<int> de(n,);
//make graph
for(auto a:pre)
{
gra[a.second].insert(a.first);
}
//calculate the indegree
for(int i=;i<n;++i)
{
for(auto a:gra[i])
de[a]++;
} for(int i=;i<n;++i)
{
int j=;
for(;j<n;++j)
{
if(de[j]==) break;
}
if(j==n) return ; //this time could not learn a new lesson and we won't make it
de[j]=-; //in case next time we will learn this lesson again
for(auto a:gra[j])
de[a]--; //any lesson take this lesson as it's prerequisites, it's indegree decrease 1
}
return ;
}
};

3 DFS

  那么DFS(深度优先搜索)是怎么样的思路呢?其实,如果这个图中存在一个环的话,那么肯定无法完成任务。那么什么情况下会完不成任务呢?也就是说1->2,2->3,3->1,也就是说形成了一个环。所以我们把原问题转化为检测图中是否有环。检测环的方法?从某个节点出发,如果又回到该节点,那么此图有环!

  我们依次从任意节点出发,遍历该节点可以到达的所有节点,我们用一个向量vector<bool> onePath来记录从该节点出发经过的节点。为了防止重复,我们用变量vector<bool> path记录所有遍历过的节点。

  上代码:

class Solution {
public:
bool canFinish(int n, vector<pair<int, int>>& pre) {
vector<unordered_set<int> > gra(n);
vector<bool> path(n,),onePath(n,);
//make graph
for(auto a:pre)
{
gra[a.second].insert(a.first);
} for(int i=;i<n;++i)
{
if(path[i]) continue;
if(dfs_circle(gra,i,path,onePath))
return ;
}
return ;
}
bool dfs_circle(vector<unordered_set<int> >& gra,int node,vector<bool>& path,vector<bool>& onePath)
{
if(onePath[node]==) return true;
path[node]=onePath[node]=;
for(auto a:gra[node])
if(dfs_circle(gra,a,path,onePath))
return ;
onePath[node]=;
return ;
}
};

  这种转化问题的解法真得很巧妙!太赞!

  另外,对于很多条件有交叉的题目,就考虑用图论方法解决,尤其是DFS和BFS算法。

BFS、DFS与选课问题(拓扑排序)的更多相关文章

  1. bfs+dfs乱搞+类似拓扑排序——cf1182D

    代码不知道上了多少补丁..终于过了 用类似拓扑排序的办法收缩整棵树得到x,然后找到x直连的最远的和最近的点 只有这三个点可能是根,依次判一下即可 另外题解的第一种方法时找直径,然后判两端点+重心+所有 ...

  2. 强大的dfs(用处1——拓扑排序【xdoj1025】,用处二——求强联通分量【ccf高速公路】)当然dfs用处多着咧

    xdoj 1025 亮亮最近在玩一款叫做“梦想庄园”的经营游戏.在游戏中,你可以耕种,养羊甚至建造纺织厂. 如果你需要制造衣服,你首先得有布匹和毛线.布匹由棉花纺织而成:毛线由羊毛制成,而羊需要饲料才 ...

  3. Ordering Tasks(拓扑排序+dfs)

    Ordering Tasks John has n tasks to do. Unfortunately, the tasks are not independent and the executio ...

  4. 拓扑排序入门详解&&Educational Codeforces Round 72 (Rated for Div. 2)-----D

    https://codeforces.com/contest/1217 D:给定一个有向图,给图染色,使图中的环不只由一种颜色构成,输出每一条边的颜色 不成环的边全部用1染色 ps:最后输出需要注意, ...

  5. 图论相关知识(DFS、BFS、拓扑排序、最小代价生成树、最短路径)

    图的存储 假设是n点m边的图: 邻接矩阵:很简单,但是遍历图的时间复杂度和空间复杂度都为n^2,不适合数据量大的情况 邻接表:略微复杂一丢丢,空间复杂度n+m,遍历图的时间复杂度为m,适用情况更广 前 ...

  6. BFS (1)算法模板 看是否需要分层 (2)拓扑排序——检测编译时的循环依赖 制定有依赖关系的任务的执行顺序 djkstra无非是将bfs模板中的deque修改为heapq

    BFS模板,记住这5个: (1)针对树的BFS 1.1 无需分层遍历 from collections import deque def levelOrderTree(root): if not ro ...

  7. hdu1532 用BFS求拓扑排序

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1285 题目给出一些点对之间的先后顺序,要求给出一个字典序最小的拓扑排列.对于拓扑排序的问题,我们有DF ...

  8. 拓扑排序详解(梅开二度之dfs版按字典序输出拓扑路径+dfs版输出全部拓扑路径

    什么是拓扑排序? 先穿袜子再穿鞋,先当孙子再当爷.这就是拓扑排序! 拓扑排序说白了其实不太算是一种排序算法,但又像是一种排序(我是不是说了个废话qwq) 他其实是一个有向无环图(DAG, Direct ...

  9. ACM/ICPC 之 拓扑排序+DFS(POJ1128(ZOJ1083)-POJ1270)

    两道经典的同类型拓扑排序+DFS问题,第二题较第一题简单,其中的难点在于字典序输出+建立单向无环图,另外理解题意是最难的难点,没有之一... POJ1128(ZOJ1083)-Frame Stacki ...

  10. 百度之星热身赛-1001(dfs拓扑排序)

    题意:作为年度优秀魔法学员的奖赏,哈利得到了一台具有魔力的计算机.这台计算机一旦开始处理某个任务,就会一直处理到这个任务结束为止(所以你可以认为它是单线程的).有一天,这台计算机得到了n个任务要处理, ...

随机推荐

  1. 解决未能加载文件或程序集'WebGrease‘的问题

    在多个视图中,如果有使用共用的样式代码,可以把它们移至CSS文件中去.今天Insus.NET就举例一个例子来说明.比如前2篇中<ASP.NET MVC图片管理(上传,预览与显示)>http ...

  2. Redis系列四之复制

    一.复制基本配置与演示 为了避免单点故障,Redis提供了复制功能,可以实现自动同步的过程. 1.配置 同步后的数据分为两类:一类是主数据库(master),一类是从数据库(slave).主数据库可以 ...

  3. CodeSnippet.info 开源说明 和 环境搭建 (第一版)

    Github开源声明 本网站的代码开源,开源的目的如下 技术分享 希望业内同行贡献代码 希望能够让网站更加安全 开源地址: CodeSnippet开源地址 关于代码贡献 任何人都可以贡献代码,一般在 ...

  4. EntityFramework中几种更改数据的方式

    首先声明个实体类,该实体类是EntityFrameWork自动生成的,对应数据表Test结构如下 public partial class Test { public int Id{ get; set ...

  5. mysql: see all open connections to a given database?

    SHOW PROCESSLIST or  show status where `variable_name` = 'Threads_connected';

  6. 袋鼠过河---DP

    题目:一只袋鼠要从河这边跳到河对岸,河很宽,但是河中间打了很多桩子,每隔一米就有一个,每个桩子上都有一个弹簧,袋鼠跳到弹簧上就可以跳的更远,每个弹簧力量不同,用一个数字代表它的力量,如果弹簧力量为5, ...

  7. [javaSE] 注解-JDK中的注解

    java中的常见注解 jdk自带注解:@Override 覆盖  @Deprecated 过期  @Suppvisewarnings 压制警告 package com.tsh.ano; public ...

  8. MAC 卸载 mysql

    怎样才能完全卸载mysql和删除它从你的Mac(包括所有数据库)执行以下操作: 打开终端窗口 使用mysqldump备份你的数据库将文本文件! 停止数据库服务器 sudo rm /usr/local/ ...

  9. css 属性选择器

    css2的属性选择器 1.[class~="flower"]:选中有flower的class class="flower ss" class="ss ...

  10. Bootstrap源码分析之transition、affix

    一.Transition(过滤) 作为一个基础支持的组件,被其他组件多次引用.实现根据浏览器支持transition的能力,然后绑定动画的结束事件:首先:创建一个Element:然后:迭代查看此元素支 ...