表示一个图有两种标准方法:图邻接表和图的邻接矩阵;
邻接矩阵就是用数组(二维)表示图。具体可以看下面例子。当然,这种情况很容易造成空间浪费,所以很多人进行空间优化,甚至是邻接表的方式。
邻接表咋是数组嵌套链表,这样会比邻接矩阵省不少空间,但对无向图来说,依旧会浪费一半的空间。
在本文中,使用 G=(V,E)
表示一个图,V表示顶点,E表示边。
深度优先搜索属于图算法的一种,英文缩写为DFS即Depth First Search.其过程简要来说是对每一个可能的分支路径深入到不能再深入为止,而且每个节点只能访问一次.
对于深度优先的流程来说,大致可以认为是这样的操作:从初始点按照一个方向搜索,直到这个方向到了尽头再回退去其他路径;他的执行像是一个栈,也像是递归;也很像是二叉树的前序遍历;
DFS(G)
for each vertex u belongs to G.V
u.color = WHITE
u.pi = NIL
time = 0
// 这里之所以做循环呢,是因为怕图是非连通图,可能会有遗漏的顶点
for each vertex u belongs to G.V
if u.color == WHITE
DFS-VISIT(G, u)
DFS-VISIT(G, u)
time = time+1
u.d = time
u.color = GRAY
for each v belongs to G.Adj[u] // 处理 u 的所有邻接点 v
if v.color == WHITE
v.pi = u
DFS-VISIT(G, v)
u.color = BLACK
time - time+1
u.f = time
BFS,其英文全称是Breadth First Search。 BFS并不使用经验法则算法。从算法的观点,所有因为展开节点而得到的子节点都会被加进一个先进先出的队列中。一般的实验里,其邻居节点尚未被检验过的节点会被放置在一个被称为 open 的容器中(例如队列或是链表),而被检验过的节点则被放置在被称为 closed 的容器中。(open-closed表)
Prim最小生成树算法和Dijkstra单源最短路径算法都采用了类似的算法。
个人理解,BFS算法在操作上非常像二叉树的层次遍历,都需要有一个队列辅助保存最近的顶点。
在给定图G=(V, E)
和特定顶点s
是,广度优先搜索搜索图中的边,期望发现s
能到达的所有顶点,并计算s
到所有顶点的距离,该算法同时能生成一棵根为s
、包括所有s
可到达顶点的广度优先树。对于可以从s
到达的任意顶点v
,广度优先树中从s
到v
的路径对应于图G
中的一条最短路径,即包含最少边数的路径。
BFS(G, s)
for each vertex u belongs to G.V - {s}
u.color = white // 未遍历过的顶点是白色
u.d = INT_MAX
u.pi = NIL
s.color = GRAY
s.d = 0
s.pi = NIL
Q = {}
ENQUEUE(Q, s) // 入队
while(Q != {})
u = DEQUEUE(Q) // 出队
for each v belongs to G.Adj[u] // 顶点u的所有邻接点
if v.color == WHITE
v.color = GRAY
v.d = u.d + 1
v.pi = u
ENQUEUE(Q, v) // 将处理过的 v 入队
u.color = BLACK // 遍历过的顶点是黑色
通过深度优先遍历解决:如果一条深度有限遍历的路径中遇到了之前访问过的结点(给每个访问的结点做标记),则证明图中存在环;
因为采用邻接矩阵存储,一般至少需要将矩阵中元素的一半给过一下,由于矩阵元素个数为n^2, 因此时间复杂度就是O(n^2)。如果采用邻接表存储,则只存储了边结点(e条边,无向图是2e条边),加上表头结点为n(也就是顶点个数),因此时间复杂度为O(n+e)。
方法是重复寻找一个入度为0的顶点,将该顶点从图中删除(即放进一个队列里存着,这个队列的顺序就是最后的拓扑排序,具体见程序),并将该结点及其所有的出边从图中删除(即该结点指向的结点的入度减1),最终若图中全为入度为1的点,则这些点至少组成一个回路。
采用邻接矩阵存储时,遍历二维数组,求各顶点入度的时间复杂度是O(n^2)。 遍历所有结点,找出入度为0的结点的时间复杂度是O(n)。对于n个入度为0的结点,删除他们的出边的复杂度为O(n^2)。 所以总的复杂度为O(n^2)。
对于邻接表,遍历所有边,求各顶点入度的时间复杂度是O(e),即边的个数。遍历所有结点,找出入度为0的结点的时间复杂度是O(n),即顶点的个数。遍历所有边,删除入度为0的结点的出边的复杂度为O(e),即边的个数。所以总的时间复杂度是O(n+e)。
prim 算法又称为最小生成树算法。
从单一顶点开始,prim算法按照以下步骤逐步扩大树中所含顶点的数目,直到遍及连通图的所有顶点。
啥叫拓扑排序?
将有向图中的顶点以线性方式进行排序。即对于任何连接自顶点u到顶点v的有向边uv,在最后的排序结果中,顶点u总是在顶点v的前面。
拓扑排序可以基于 DFS 实现:
L ← Empty list that will contain the sorted nodes
S ← Set of all nodes with no outgoing edges
for each node n in S do
visit(n)
function visit(node n)
if n has not been visited yet then
mark n as visited
for each node m with an edgefrom m to ndo
visit(m)
add n to L
如题:给的图只有一个结点,让你深度拷贝一个完整的图;
思路:
class Solution {
public:
UndirectedGraphNode *cloneGraph(UndirectedGraphNode *node) {
if(node == NULL)
return NULL;
unordered_map<int, UndirectedGraphNode *> m; //origin graph
queue<UndirectedGraphNode *> q;
q.push(node);
while(!q.empty())
{
UndirectedGraphNode *tmp = q.front();
q.pop();
if(m.find(tmp->label) == m.end())
m.insert(pair<int, UndirectedGraphNode *>(tmp->label, tmp));
for(int i=0;i < tmp->neighbors.size();i++)
if(m.find(tmp->neighbors[i]->label) == m.end())
q.push(tmp->neighbors[i]);
}
unordered_map<int, UndirectedGraphNode *> clone; // clone graph
unordered_map<int, UndirectedGraphNode *>::iterator it = m.begin();
// Step 1: create new node
for(;it != m.end(); it++)
{
UndirectedGraphNode * temp = new UndirectedGraphNode((*it).first);
clone.insert(pair<int, UndirectedGraphNode *>((*it).first, temp));
}
// Step 2: fill the neighbors
it = m.begin();
for(;it != m.end(); it++)
{
for(int i=0;i<(*it).second->neighbors.size();i++)
{
clone[(*it).first]->neighbors.push_back( clone[(*it).second->neighbors[i]->label] );
}
}
return clone[node->label];
}
}