类型一:邻接表

题目一:员工的重要性

题目描述

给定一个保存员工信息的数据结构,它包含了员工唯一的id重要度直系下属的id

比如,员工1是员工2的领导,员工2是员工3的领导。他们相应的重要度为15, 10, 5。那么员工1的数据结构是[1, 15, [2]],员工2的数据结构是[2, 10, [3]],员工3的数据结构是[3, 5, []]。注意虽然员工3也是员工1的一个下属,但是由于并不是直系下属,因此没有体现在员工1的数据结构中。

现在输入一个公司的所有员工信息,以及单个员工id,返回这个员工和他所有下属的重要度之和。

示例 1:

  1. 输入: [[1, 5, [2, 3]], [2, 3, []], [3, 3, []]], 1
  2. 输出: 11
  3. 解释:
  4. 员工1自身的重要度是5,他有两个直系下属23,而且23的重要度均为3。因此员工1的总重要度是 5 + 3 + 3 = 11

注意:

  1. 一个员工最多有一个直系领导,但是可以有多个直系下属
  2. 员工数量不超过2000。

思路:DFS

  【注意点:应使用局部变量(weight)记录结果,不能使用全局变量】

代码:

  1. """
  2. # Employee info
  3. class Employee(object):
  4. def __init__(self, id, importance, subordinates):
  5. self.id = id
  6. self.importance = importance
  7. self.subordinates = subordinates
  8. """
  9. class Solution(object):
  10. def getImportance(self, employees, id):
  11. """
  12. :type employees: Employee
  13. :type id: int
  14. :rtype: int
  15. """
  16. if not employees:
  17. return 0
  18. ##局部变量
  19. weight = 0
  20. for item in employees:
  21. if item.id == id:
  22. weight += item.importance
  23. for j in item.subordinates:
  24. weight += self.getImportance(employees, j)
  25. return weight

题目二:钥匙和房间

N 个房间,开始时你位于 0 号房间。每个房间有不同的号码:0,1,2,...,N-1,并且房间里可能有一些钥匙能使你进入下一个房间。

在形式上,对于每个房间 i 都有一个钥匙列表 rooms[i],每个钥匙 rooms[i][j][0,1,...,N-1] 中的一个整数表示,其中 N = rooms.length。 钥匙 rooms[i][j] = v 可以打开编号为 v 的房间。

最初,除 0 号房间外的其余所有房间都被锁住。

你可以自由地在房间之间来回走动。

如果能进入每个房间返回 true,否则返回 false

示例 1:

  1. 输入: [[1],[2],[3],[]]
  2. 输出: true
  3. 解释:
  4. 我们从 0 号房间开始,拿到钥匙 1
  5. 之后我们去 1 号房间,拿到钥匙 2
  6. 然后我们去 2 号房间,拿到钥匙 3
  7. 最后我们去了 3 号房间。
  8. 由于我们能够进入每个房间,我们返回 true

示例 2:

  1. 输入:[[1,3],[3,0,1],[2],[0]]
  2. 输出:false
  3. 解释:我们不能进入 2 号房间。

提示:

  1. 1 <= rooms.length <= 1000
  2. 0 <= rooms[i].length <= 1000
  3. 所有房间中的钥匙数量总计不超过 3000

思路:DFS

每个房间找钥匙,找到则到下一个钥匙对应的房间DFS。采用一个visited列表存储是否到达过这个房间,最后如果所有房间都达到过则返回True。

代码

  1. def canVisitAllRooms(rooms):
  2. if not rooms:
  3. return True
  4. n = len(rooms)
  5. visited = [False] * n
  6. visited[0] = True
  7. def dfs(rooms,keys,visited):
  8. for key in keys:
  9. if not visited[key]:
  10. visited[key] = True
  11. dfs(rooms,rooms[key],visited)
  12. dfs(rooms,rooms[0],visited)
  13. return all(visited)

类型二:DAG拓扑排序

题目一:课程安排207

题目二:课程安排210

现在你总共有 n 门课需要选,记为 0n-1

在选修某些课程之前需要一些先修课程。 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们: [0,1]

给定课程总量以及它们的先决条件,返回你为了学完所有课程所安排的学习顺序。

可能会有多个正确的顺序,你只要返回一种就可以了。如果不可能完成所有课程,返回一个空数组。

示例 1:

  1. 输入: 2, [[1,0]]
  2. 输出: [0,1]
  3. 解释: 总共有 2 门课程。要学习课程 1,你需要先完成课程 0。因此,正确的课程顺序为 [0,1]

示例 2:

  1. 输入: 4, [[1,0],[2,0],[3,1],[3,2]]
  2. 输出: [0,1,2,3] or [0,2,1,3]
  3. 解释: 总共有 4 门课程。要学习课程 3,你应该先完成课程 1 和课程 2。并且课程 1 和课程 2 都应该排在课程 0 之后。
  4. 因此,一个正确的课程顺序是 [0,1,2,3] 。另一个正确的排序是 [0,2,1,3]

说明:

  1. 输入的先决条件是由边缘列表表示的图形,而不是邻接矩阵。详情请参见图的表示法
  2. 你可以假定输入的先决条件中没有重复的边。

提示:

  1. 这个问题相当于查找一个循环是否存在于有向图中。如果存在循环,则不存在拓扑排序,因此不可能选取所有课程进行学习。
  2. 通过 DFS 进行拓扑排序 - 一个关于Coursera的精彩视频教程(21分钟),介绍拓扑排序的基本概念。
  3. 拓扑排序也可以通过 BFS 完成。

思路:DFS

  1、建立图

  2、循环n次,每次是遍历一个节点是否已经visited且合法地加入path中了,如果False不合法则直接返回【】。

  3、遍历一个节点时会将其后面的所有子节点都处理掉。

代码

  1. from collections import defaultdict
  2. def findPath(n,arr):
  3. if n == 0:
  4. return []
  5. graph = defaultdict(list)
  6. for u , v in arr:
  7. graph[v].append(u)
  8. # 0为Unkown,1为visiting,2为visited
  9. path = []
  10. visited = [0] * n
  11. for i in range(n):
  12. if not DFS(graph,visited,path,i):
  13. return []
  14. return path[::-1]
  15. def DFS(graph,visited,path,i):
  16. ####i节点:其正在遍历,但它的子节点的子节点也是它,表示产生了有环,则return FALSE
  17. if visited[i] == 1: return False
  18. ####i节点 :已经遍历过,后面已经没有节点了,return true
  19. elif visited[i] == 2:return True
  20. ####表示正在遍历i节点
  21. visited[i] = 1
  22. for j in graph[i]:
  23. if not DFS(graph,visited,path,j):
  24. return False
  25. path.append(i)
  26. visited[i] = 2
  27. return True
  28. n = 5
  29. arr = [[1,0],[2,0],[3,1],[3,2],[4,0]]
  30. print(findPath(n,arr))

题目三:解题报告,连除----399

已经给出了某些变量的比值,求新的变量的比值。如果这个变量没有出现过,或者不可到达,那么返回-1.

DFS思路 

题目中给了顶点和顶点之间的关系,其实就是制定了这个图的样子。然后要求的新的比值其实就是从一个顶点到达另外一个顶点的路径,并且把这条路径上所有的权重相乘。

注意,如果a/b=3,那么从a到b是3,那么从b到a是1/3.

既然是从一个顶点出发到达另外一个顶点,所以应该是dfs解决的问题。

原文:https://blog.csdn.net/fuxuemingzhu/article/details/82591165

  1、建立图  {a:{b : 2.0} 、b:{a:1 /2.0,c:3.0}、c:{b:1/3.0}}

  2、不在图中则返回-1  

  3、在图中,x == y,返回1,x != y,返回x到y的拓扑排序的权重相乘值。

代码:

  

  1. from collections import defaultdict
  2. def solveque(arr ,values , que):
  3. if not arr:
  4. return [-1] * len(que)
  5. if not que:
  6. return []
  7. graph = defaultdict(dict)
  8. for (x,y) , value in zip(arr,values):
  9. graph[x][y] = value
  10. graph[y][x] = 1/value if value else 0
  11. for x,y in que:
  12. if x in graph and y in graph:
  13. return dfs(graph,x,y,set())
  14. else:
  15. return -1.0
  16. def dfs(graph,x,y,visited):
  17. if x == y:
  18. return 1.0
  19. visited.add(x)
  20. for k in graph[x]:
  21. if k in visited:continue
  22. visited.add(k)
  23. d = dfs(graph,k,y,visited)
  24. if d > 0:
  25. return d*graph[x][k]
  26. return -1.0
  27. arr = [['a','b'],['b','c']]
  28. values = [2.0,3.0]
  29. que = [['a','c'],['a','v']]
  30. print(solveque(arr ,values , que))

图的遍历---DFS的更多相关文章

  1. 图的遍历DFS

    图的遍历DFS 与树的深度优先遍历之间的联系 树的深度优先遍历分为:先根,后根 //树的先根遍历 void PreOrder(TreeNode *R){ if(R!=NULL){ visit(R); ...

  2. 图的遍历——DFS(矩形空间)

    首先,这里的图不是指的我们一般所说的图结构,而是大小为M*N的矩形区域(也可以看成是一个矩阵).而关于矩形区域的遍历问题经常出现,如“寻找矩阵中的路径”.“找到矩形区域的某个特殊点”等等之类的题目,在 ...

  3. 图的遍历——DFS和BFS模板(一般的图)

    关于图的遍历,通常有深度优先搜索(DFS)和广度优先搜索(BFS),本文结合一般的图结构(邻接矩阵和邻接表),给出两种遍历算法的模板 1.深度优先搜索(DFS) #include<iostrea ...

  4. 图的遍历——DFS

    原创 图的遍历有DFS和BFS两种,现选用DFS遍历图. 存储图用邻接矩阵,图有v个顶点,e条边,邻接矩阵就是一个VxV的矩阵: 若顶点1和顶点5之间有连线,则矩阵元素[1,5]置1,若是无向图[5, ...

  5. 图的遍历——DFS(邻接矩阵)

    递归 + 标记 一个连通图只要DFS一次,即可打印所有的点. #include <iostream> #include <cstdio> #include <cstdli ...

  6. 图的遍历(DFS、BFS)

    理论: 深度优先搜索(Depth_Fisrst Search)遍历类似于树的先根遍历,是树的先根遍历的推广: 广度优先搜索(Breadth_First Search) 遍历类似于树的按层次遍历的过程: ...

  7. 16.boost图深度优先遍历DFS

    #include <iostream> #include <boost/config.hpp> //图(矩阵实现) #include <boost/graph/adjac ...

  8. 图的遍历[DFS][BFS]

    #include<iostream> #include<iostream> #include<cstring> #include<queue> #inc ...

  9. 图的数据结构的实现与遍历(DFS,BFS)

    //图的存储结构:const int MAXSIZE = 10;//邻接矩阵template<class T>class MGraph {public:    MGraph(T a[], ...

随机推荐

  1. N天学习一个linux命令之du

    用途 统计文件或者目录占用硬盘空间大小 用法 du [OPTION] [FILE]du [OPTION] --files0-from=F 常用参数 -a, --all统计所有文件,不仅仅是目录 -b, ...

  2. 快速傅立叶变换&HDU 1402

    参考http://www.cnblogs.com/v-July-v/archive/2011/08/13/2214132.html <算导> 那么,更快速的多项式乘法就依赖于能否把一个系数 ...

  3. C++对象模型——对象复制语意学 (Object Copy Semantics)(第五章)

    5.3    对象复制语意学 (Object Copy Semantics) 当设计一个 class,并以一个 class object指定给 class object时,有三种选择:     1.什 ...

  4. springmvc 解析xml数据

    springmvc 解析xml数据 http://blog.csdn.net/zhi_jun/article/details/37925475

  5. 在NSUserDefaults中保存自己定义的对象

    在iOS开发中.须要用到一些回调值(从A到B,从B返回时把B中的值带回A中).事实上方法也非常多(delegate,block.nsuserdefaults等).我想用NSUserDefaults回调 ...

  6. SpringMVC案例2----基于spring2.5的注解实现

    和上一篇一样,首先看一下项目结构和jar包 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvYmVuamFtaW5fd2h4/font/5a6L5L2T/fo ...

  7. [java设计模式]之单例模式

    -------------------此部分比較深入地解说了单例模式,原文链接已给出.兴许将涉及一些常见面试问题--------------------------- 原文地址:http://www. ...

  8. Linux下,安装配置Weblogic

    环境说明 系统 -- Linux RHEL5 32bit 环境 -- 局域网中在192.168.0.140(windows)通过xshell连接服务器 软件 -- 1.JDK:1.5.0_15  2. ...

  9. C#格式化年月日截取

     //if (bm.Name == "DateYear") //年                 //{                 //    bm.Select();   ...

  10. ROS-URDF-Xacro

    前言:Xacro是一种宏语言,允许代码复用,使用Xacro可以减少URDF文件中的代码量. 参考自:http://wiki.ros.org/urdf/Tutorials/Using%20Xacro%2 ...