《算法导论》学习总结 — XX.第22章 图的基本算法
BFS(广搜):
广搜就是广度优先搜索,根据名字可以知道,是通过广度来遍历图,也就是层次遍历吧。
在这里以及下面的DFS(深搜),都用到了颜色WHITE,GRAY,BLACK,不过作用不同,具体分别再分析。
在BFS中,WHITE,GRAY,BLACK这三色是用来记录一个点是否被搜到,以及是否它的邻接点都是灰色。(具体见P324倒数第2段)。
P326 的图22-3是个经典的图,看了此图基本就知道BFS是干嘛的了。
在图22-3中,因为他是用字母表示的,我把各点定义为顺时针从标号1开始,于是r点是1号,源点s是2号。
这是我写的具体实现(C/C++):
/*
* Introduction to Algorithms
* Chapter 22 --- BFS
* Tanky Woo @ www.WuTianQi.com
* 2012.1.6
*/
#include
#include
#define WHITE 0
#define GRAY 1
#define BLACK 2
#define MAX 999999
#define NIL 0 using namespace std; int node;
int G[100][100];
int color[100];
int dist[100];
int prev[100]; void BFS(int source)
{
for(int i=1; i<=node; ++i)
{
color[i] = WHITE;
dist[i] = MAX;
prev[i] = NIL;
}
color = GRAY;
dist = 0;
prev = NIL; queue que;
que.push(source); while(!que.empty())
{
int no = que.front();
que.pop();
for(int i=1; i<=node; ++i)
if(G[no][i]==1 && color[i]==WHITE)
{
color[i] = GRAY;
dist[i] = dist[no] + 1;
prev[i] = no;
que.push(i);
}
color[no] = BLACK;
} } void printPath(int source, int dest)
{
if(source == dest)
cout << source;
else if(prev[dest] == NIL)
cout << "No path from " << source << " to " << dest << " exist\n";
else
{
printPath(source, prev[dest]);
cout << " " << dest;
} } int main()
{
freopen("input.txt", "r", stdin); memset(G, 0, sizeof(G));
cout << "Input the number of the node:\n";
cin >> node;
cout << "Input the Graph:\n";
for(int i=1; i<=node; ++i)
for(int j=1; j<=node; ++j)
cin >> G[i][j];
BFS(2); cout << dist[4] << endl;
cout << "The path: ";
printPath(2, 4); }
这是input.txt的内容:
8
0 1 0 0 0 0 0 1
1 0 0 0 0 0 1 0
0 0 0 1 0 1 1 0
0 0 1 0 1 1 0 0
0 0 0 1 0 1 1 0
0 0 1 1 1 0 0 0
0 1 1 0 1 0 0 0
1 0 0 0 0 0 0 0
可以看到结果,源点(2号)到u点(4号)的距离是3.
源码我上传到我的网盘了:
http://www.kuaipan.cn/file/id_35241628297855049.html
DFS(深搜):
深搜既深度优先搜索,相对于BFS,它是尽可能深的去搜索一个图。
在DFS中,仍然用到了WHITE, GRAY,BLACK,但是它们的用处和BFS有些不同,这里他们是用来表示时间戳,因为DFS会有回溯,所以也就有第一次和第二次搜到。(具体见P330面)
看P331面图22-4。
具体实现(C/C++):
/*
* Introduction to Algorithms
* Chapter 22 --- DFS
* Tanky Woo @ www.WuTianQi.com
* 2012.1.6
*/ #include
#include
#define WHITE 0
#define GRAY 1
#define BLACK 2
#define NIL 0 using namespace std; int node;
int G[100][100];
int color[100];
int fir[100], sec[100]; // 用来记录前后时间戳
int prev[100];
int time; // time用来记录时间戳 void DFS_VISIT(int no)
{
color[no] = GRAY; // 当点no第一次被搜到
++time;
fir[no] = time;
for(int i=1; i<=node; ++i)
{
if(G[no][i]==1 && color[i]==WHITE)
{
prev[i] = no;
DFS_VISIT(i);
}
}
color[no] = BLACK; // 当点no深搜完回溯时再次搜到
sec[no] = ++time; } void DFS()
{
for(int i=1; i<=node; ++i)
{
if(color[i] == WHITE)
prev[i] = NIL;
}
time = 0;
for(int i=1; i<=node; ++i)
{
if(color[i] == WHITE)
DFS_VISIT(i);
}
} int main()
{
freopen("input.txt", "r", stdin); memset(G, 0, sizeof(G));
cout << "Input the number of the node:\n";
cin >> node;
cout << "Input the Graph:\n";
for(int i=1; i<=node; ++i)
for(int j=1; j<=node; ++j)
cin >> G[i][j]; DFS();
cout << "点v(2号)的first:" << fir[2] << " 和second:" << sec[2] << endl;
}
这是input.txt的内容:
6
0 1 0 0 0 1
0 0 0 0 1 0
0 0 0 1 1 0
0 0 0 1 0 0
0 0 0 0 0 1
0 1 0 0 0 0
可以看到结果,点v(2号)的first和second分别是2和7
源码我上传到我的网盘了:
http://www.kuaipan.cn/file/id_35241628297855052.html
(拓扑排序下次再写吧)
小结:
DFS和BFS是图算法的根本,但是刚开始了解可能不太习惯,建议找一些相应的题目来试试手。可以在我Blog搜BFS和DFS,可以找到我曾经做过的题目。
《算法导论》学习总结 — XX.第22章 图的基本算法的更多相关文章
- (搬运)《算法导论》习题解答 Chapter 22.1-1(入度和出度)
(搬运)<算法导论>习题解答 Chapter 22.1-1(入度和出度) 思路:遍历邻接列表即可; 伪代码: for u 属于 Vertex for v属于 Adj[u] outdegre ...
- 算法导论学习---红黑树具体解释之插入(C语言实现)
前面我们学习二叉搜索树的时候发如今一些情况下其高度不是非常均匀,甚至有时候会退化成一条长链,所以我们引用一些"平衡"的二叉搜索树.红黑树就是一种"平衡"的二叉搜 ...
- 《算法导论》学习总结 — XX.第23章 最小生成树
一.什么叫最小生成树 一个无向连通图G=(V,E),最小生成树就是联结所有顶点的边的权值和最小时的子图T,此时T无回路且连接所有的顶点,所以它必须是棵树. 二.为什么要研究最小生成树问题 <算法 ...
- 算法导论学习-Dynamic Programming
转载自:http://blog.csdn.net/speedme/article/details/24231197 1. 什么是动态规划 ------------------------------- ...
- 算法导论学习之线性时间求第k小元素+堆思想求前k大元素
对于曾经,假设要我求第k小元素.或者是求前k大元素,我可能会将元素先排序,然后就直接求出来了,可是如今有了更好的思路. 一.线性时间内求第k小元素 这个算法又是一个基于分治思想的算法. 其详细的分治思 ...
- 《算法导论》习题解答 Chapter 22.1-5(求平方图)
一.邻接矩阵实现 思路:如果是邻接矩阵存储,设邻接矩阵为A,则A*A即为平方图,只需要矩阵相乘即可: 伪代码: for i=1 to n for j=1 to n for k=1 to n resul ...
- 算法导论学习-RED-BLACK TREE
1. 红黑树(RED-BLACK TREE)引言: ------------------------------------- 红黑树(RBT)可以说是binary-search tree的非严格的平 ...
- 《算法导论》习题解答 Chapter 22.1-8(变换邻接表的数据结构)
一般散列表都与B+树进行比较,包括在信息检索中也是. 确定某条边是否存在需要O(1). 不足: (1)散列冲突. (2)哈希函数需要不断变化以适应需求. 另外:B+树.(见第18章) 与散列表相比的不 ...
- 《算法导论》习题解答 Chapter 22.1-6(求universal sink 通用汇点)
思路:设置两个游标i指向行,j指向列,如果arr[i][j]==1,则i=max{i+1,j},j++:如果arr[i][j]==0,则j=max{i+1,j+1}. 伪代码: has_univers ...
随机推荐
- NCS8801S芯片RGB/LVDS转EDP功能简介
NCS8801S RGB/LVDS-to-eDP Converter (1/2/4-lane eDP) Features --Embedded-DisplayPort (eDP) Output 1/2 ...
- Spring Cloud Netflix多语言/非java语言支持之Spring Cloud Sidecar
Spring Cloud Netflix多语言/非java语言支持之Spring Cloud Sidecar 前言 公司有一个调研要做,调研如何将Python语言提供的服务纳入到Spring Clou ...
- Mac上安装openCV(Java版本)
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt410 Install and use OpenCV 3.0 on Mac O ...
- 【ACM小白成长撸】--贪婪法解硬币找零问题
question:假设有一种货币,它有面值为1分.2分.5分和1角的硬币,最少需要多少个硬币来找出K分钱的零钱.按照贪婪法的思想,需要不断地使用面值最大的硬币.如果找零的值小于最大的硬币值,则尝试第二 ...
- Cobbler批量部署CentOS
简介 Cobbler是一个快速网络安装linux的服务,而且在经过调整也可以支持网络安装windows.该工具使用python开发,小巧轻便(才15k行python代码),使用简单的命令即可完成PXE ...
- 软工+C(2017第1期) 题目设计、点评和评分
// 下一篇:分数和checklist 如何设计题目 教学中的一个问题是老师出题太简单了,题目设计一开始上来就不紧凑,我认为一个好的课程应该上来就给你紧凑感,而不是先上来"轻松2-3周&qu ...
- 【Beta】 第五次Daily Scrum Meeting
一.本次会议为第五次meeting会议 二.时间:10:00AM-10:20AM 地点:陆大楼 三.会议站立式照片 四.今日任务安排 成员 昨日任务 今日任务 林晓芳 帮助完善登录界面 对目前完成的模 ...
- 【Beta】 第一次Daily Scrum Meeting
一.本次会议为第一次meeting会议 二.时间:20::0AM-20:50AM 地点:宿舍楼下 三.会议站立式照片 四.今日任务安排 成员 昨日任务 今日任务 林晓芳 对已完成的功能进行进一步测 ...
- 201521123079《java程序设计》第8周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结集合与泛型相关内容. 2. 书面作业 本次作业题集集合 1.List中指定元素的删除(题目4-1) 1.1 实验总结 这题主要是对函 ...
- 201521123033《Java程序设计》第5周学习总结
1. 本周学习总结 1.1 尝试使用思维导图总结有关多态与接口的知识点. 参考资料: 百度脑图 XMind 2. 书面作业 作业参考文件下载 1.代码阅读:Child压缩包内源代码 1.1 com.p ...