一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - C/C++ - C++深度優先搜索的實現方法

C++深度優先搜索的實現方法

2021-01-28 14:50C++教程網 C/C++

這篇文章主要介紹了C++深度優先搜索的實現方法,是數據結構中非常重要的一種算法,需要的朋友可以參考下

本文實例講述了圖的遍歷中深度優先搜索C++實現方法,是一種非常重要的算法,具體實現方法如下:

首先,圖的遍歷是指從圖中的某一個頂點出發,按照某種搜索方法沿著圖中的邊對圖中的所有頂點訪問一次且僅訪問一次。注意到樹是一種特殊的圖,所以樹的遍歷實際上也可以看作是一種特殊的圖的遍歷。圖的遍歷主要有兩種算法:廣度優先搜索(Breadth-First-Search)和深度優先搜索(Depth-First-Search)。

一、深度優先搜索(DFS)的算法思想

深度優先搜索算法所遵循的搜索策略是盡可能“深”地搜索一個圖。它的基本思想就是:首先訪問圖中某一起始頂點v,然后由v出發,訪問與v鄰接且未被訪問的任一頂點w1,再訪問與w1鄰接且未被訪問的任一頂點w2,……重復上述過程。當不能再繼續向下訪問時,依次退回到最近被訪問的頂點,若它還有鄰接頂點未被訪問過,則從該點開始繼續上述搜索過程,直到圖中所有頂點均被訪問過為止。

C++深度優先搜索的實現方法

如上圖所示,從頂點2開始深度優先遍歷圖,結果為:2,0,1,3。

二、DFS算法實現

和廣度優先搜索一樣,為了防止頂點被多次訪問,需要使用一個訪問標記數組visited[]來標記頂點是否已經被訪問過。

這里使用鄰接表表示圖。對于一個有向圖,假設從給定頂點可以訪問到圖的所有其他頂點,則DFS遞歸算法的C++代碼實現:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/*************************************************************************
  > File Name: DFS.cpp
  > Author: SongLee
 ************************************************************************/
#include<iostream>
#include<list>
using namespace std;
 
/* 圖 */
class Graph
{
  int V;                // 頂點數
  list<int> *adj;           // 鄰接表
  void DFSUtil(int v, bool visited[]); // 從頂點v深度優先遍歷
public:
  Graph(int V);            // 構造函數
  void addEdge(int v, int w);     // 向圖中添加邊
  void DFS(int v);           // 從v開始深度優先遍歷圖
};
 
/* 構造函數 */
Graph::Graph(int V)
{
  this->V = V;
  adj = new list<int>[V];
}
 
/* 添加邊,構造鄰接表 */
void Graph::addEdge(int v, int w)
{
  adj[v].push_back(w);         // 將w添加到v的鏈表
}
 
/* 從v開始深度優先遍歷 */
void Graph::DFSUtil(int v, bool visited[])
{
  // 訪問頂點v并輸出
  visited[v] = true;
  cout << v << " ";
 
  list<int>::iterator i;
 
  for(i=adj[v].begin(); i!=adj[v].end(); ++i)
    if(!visited[*i])       // 若鄰接點尚未訪問
      DFSUtil(*i, visited);   // 遞歸
}
 
/* 對圖進行深度優先遍歷,調用遞歸函數DFSUtil() */
void Graph::DFS(int v)
{
  bool *visited = new bool[V];
  for(int i=0; i<V; ++i)
    visited[i] = false;
 
  // 假設從給定頂點v可以到達圖的所有頂點
  DFSUtil(v, visited);
}
 
/* 測試 */
int main()
{
  Graph g(4);
  g.addEdge(0, 1);
  g.addEdge(0, 2);
  g.addEdge(1, 2);
  g.addEdge(2, 0);
  g.addEdge(2, 3);
  g.addEdge(3, 3);
 
  cout << "Depth First Traversal (starting from vertex 2) \n";
  g.DFS(2);
  cout << endl;
   
  return 0;
}

上面的代碼是假設從給定頂點可以訪問到圖的所有其他頂點。如果沒有這個假設,為了對圖作一個完整的深度優先遍歷,我們需要對每個頂點調用DFSUtil()。當然那之前需要先檢查頂點是否已經訪問過。所以我們只需要修改DFS()函數部分:

?
1
2
3
4
5
6
7
8
9
10
11
void Graph::DFS()
{
  bool *visited = new bool[V];
  for(int i=0; i<V; ++i)
    visited[i] = false;
   
  // 對每個頂點調用DFSUtil(),從0開始
  for(int i=0; i<V; ++i)
    if(!visited[i])
      DFSUtil(i, visited);
}

對于無向圖的深度優先搜索,只是鄰接表不一樣,其他的都是一樣的。我們只需要修改addEdge(v, w)函數:

?
1
2
3
4
5
void Graph::addEdge(int v, int w)
{
  adj[v].push_back(w);     // 將w加到v的list
  adj[w].push_back(v);
}

注意:圖的鄰接矩陣表示是唯一的,但對于鄰接表來說,如果邊的輸入次序不同,生成的鄰接表也不同。因此,對于同一個圖,基于鄰接矩陣的遍歷所得到的DFS序列和BFS序列是唯一的,基于鄰接表的遍歷所得到的DFS序列和BFS序列是不唯一的。

三、DFS算法性能分析

1 . 空間復雜度

DFS算法是一個遞歸算法,需要借助一個遞歸工作棧,故它的空間復雜度為O(|V|)。

2 . 時間復雜度

當以鄰接表存儲時,時間復雜度為O(|V|+|E|)。

當以鄰接矩陣存儲時,時間復雜度為O(|V|^2)。

延伸 · 閱讀

精彩推薦
  • C/C++C/C++經典實例之模擬計算器示例代碼

    C/C++經典實例之模擬計算器示例代碼

    最近在看到的一個需求,本以為比較簡單,但花了不少時間,所以下面這篇文章主要給大家介紹了關于C/C++經典實例之模擬計算器的相關資料,文中通過示...

    jia150610152021-06-07
  • C/C++詳解c語言中的 strcpy和strncpy字符串函數使用

    詳解c語言中的 strcpy和strncpy字符串函數使用

    strcpy 和strcnpy函數是字符串復制函數。接下來通過本文給大家介紹c語言中的strcpy和strncpy字符串函數使用,感興趣的朋友跟隨小編要求看看吧...

    spring-go5642021-07-02
  • C/C++學習C++編程的必備軟件

    學習C++編程的必備軟件

    本文給大家分享的是作者在學習使用C++進行編程的時候所用到的一些常用的軟件,這里推薦給大家...

    謝恩銘10102021-05-08
  • C/C++C語言中炫酷的文件操作實例詳解

    C語言中炫酷的文件操作實例詳解

    內存中的數據都是暫時的,當程序結束時,它們都將丟失,為了永久性的保存大量的數據,C語言提供了對文件的操作,這篇文章主要給大家介紹了關于C語言中文件...

    針眼_6702022-01-24
  • C/C++c++ 單線程實現同時監聽多個端口

    c++ 單線程實現同時監聽多個端口

    這篇文章主要介紹了c++ 單線程實現同時監聽多個端口的方法,幫助大家更好的理解和學習使用c++,感興趣的朋友可以了解下...

    源之緣11542021-10-27
  • C/C++C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下...

    青山的青6062022-01-04
  • C/C++深入理解goto語句的替代實現方式分析

    深入理解goto語句的替代實現方式分析

    本篇文章是對goto語句的替代實現方式進行了詳細的分析介紹,需要的朋友參考下...

    C語言教程網7342020-12-03
  • C/C++C語言實現電腦關機程序

    C語言實現電腦關機程序

    這篇文章主要為大家詳細介紹了C語言實現電腦關機程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    xiaocaidayong8482021-08-20
主站蜘蛛池模板: 欧美日韩高清观看一区二区 | 久久黄色精品视频 | 亚洲成年人免费网站 | 免费国产高清精品一区在线 | 粉嫩国产14xxxxx0000| 99在线视频免费 | 亚洲美日韩 | 乌克兰粉嫩摘花第一次 | 香蕉免费一区二区三区 | 国产卡一卡二卡三乱码手机 | 性色香蕉AV久久久天天网 | 无遮18禁在线永久免费观看挡 | 操儿子 | 日本草草视频在线观看 | 好男人资源大全免费观看 | 网红刘婷hd国产高清 | 久久两性视频 | 亚洲成人影院在线观看 | 日韩在线观看一区二区不卡视频 | 女人扒开下面让男人桶爽视频 | 暖暖在线日本 | 日韩网站在线观看 | 亚洲高清中文字幕一区二区三区 | 日本久久影视 | 奇米精品 | 97伊人久久精品亚洲午夜 | a在线观看欧美在线观看 | 铁牛tv 在线观看 | 黄色aaa级片 | 国产高清在线精品一区 | 亚洲AV国产国产久青草 | 洗濯屋H纯肉动漫在线观看 武侠艳妇屈辱的张开双腿 午夜在线观看免费观看 视频 | 美女扒开屁股 | 亚洲男人天堂网址 | 国产按摩系列 | 欧美亚洲另类综合 | 精品无码乱码AV | free chinese麻豆 | 国色天香社区视频免费观看3 | 国产91区| 青草视频免费观看 |