国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 學院 > 邏輯算法 > 正文

算法系列15天速成 第十四天 圖【上】

2024-09-08 23:18:41
字體:
供稿:網(wǎng)友

今天來分享一下圖,這是一種比較復雜的非線性數(shù)據(jù)結(jié)構(gòu),之所以復雜是因為他們的數(shù)據(jù)元素之間的關(guān)系是任意的,而不像樹那樣 被幾個性質(zhì)定理框住了,元素之間的關(guān)系還是比較明顯的,圖的使用范圍很廣的,比如網(wǎng)絡(luò)爬蟲,求最短路徑等等,不過大家也不要膽怯,

越是復雜的東西越能體現(xiàn)我們碼農(nóng)的核心競爭力。       

      既然要學習圖,得要遵守一下圖的游戲規(guī)則。

一: 概念

       圖是由“頂點”的集合和“邊”的集合組成。記作:G=(V,E);

<1> 無向圖

       就是“圖”中的邊沒有方向,那么(V1,V2)這條邊自然跟(V2,V1)是等價的,無向圖的表示一般用”圓括號“。

        

<2> 有向圖

       “圖“中的邊有方向,自然<V1,V2>這條邊跟<V2,V1>不是等價的,有向圖的表示一般用"尖括號"表示。

             

<3> 鄰接點

             一條邊上的兩個頂點叫做鄰接點,比如(V1,V2),(V1,V3),(V1,V5),只是在有向圖中有一個“入邊,出邊“的

       概念,比如V3的入邊為V5,V3的出邊為V2,V1,V4。

 

<4> 頂點的度

          這個跟“樹”中的度的意思一樣。不過有向圖中也分為“入度”和“出度”兩種,這個相信大家懂的。

 

<5> 完全圖

         每兩個頂點都存在一條邊,這是一種完美的表現(xiàn),自然可以求出邊的數(shù)量。

        無向圖:edges=n(n-1)/2;

        有向圖:edges=n(n-1);           //因為有向圖是有邊的,所以必須在原來的基礎(chǔ)上"X2"。

       

<6> 子圖

        如果G1的所有頂點和邊都在G2中,則G1是G2的子圖,具體不說了。

 

<7> 路徑,路徑長度和回路(這些概念還是比較重要的)

       路徑:        如果Vm到Vn之間存在一個頂點序列。則表示Vm到Vn是一條路徑。

       路徑長度:  一條路徑中“邊的數(shù)量”。

       簡單路徑:  若一條路徑上頂點不重復出現(xiàn),則是簡單路徑。

       回路:       若路徑的第一個頂點和最后一個頂點相同,則是回路。

       簡單回路:  第一個頂點和最后一個頂點相同,其它各頂點都不重復的回路則是簡單回路。

 

<8> 連通圖和連通分量(針對無向圖而言的)

       連通圖:     無向圖中,任意兩個頂點都是連通的則是連通圖,比如V1,V2,V4之間。

       連通分量:  無向圖的極大連通子圖就是連通分量,一般”連通分量“就是”圖“本身,除非是“非連通圖”,

                       如下圖就是兩個連通分量。

           

<9> 強連通圖和強連通分量(針對有向圖而言)

        這里主要注意的是“方向性“,V4可以到V3,但是V3無法到V4,所以不能稱為強連通圖。

       

<10> 網(wǎng)

        邊上帶有”權(quán)值“的圖被稱為網(wǎng)。很有意思啊,呵呵。

 

二:存儲

     圖的存儲常用的是”鄰接矩陣”和“鄰接表”。

     鄰接矩陣: 手法是采用兩個數(shù)組,一個一維數(shù)組用來保存頂點信息,一個二維數(shù)組來用保存邊的信息,

                    缺點就是比較耗費空間。

     鄰接表:   改進后的“鄰接矩陣”,缺點是不方便判斷兩個頂點之間是否有邊,但是相比節(jié)省空間。

 

三: 創(chuàng)建圖

     這里我們就用鄰接矩陣來保存圖,一般的操作也就是:①創(chuàng)建,②遍歷

復制代碼 代碼如下:

#region 鄰接矩陣的結(jié)構(gòu)圖
    /// <summary>
/// 鄰接矩陣的結(jié)構(gòu)圖
/// </summary>
    public class MatrixGraph
    {
        //保存頂點信息
        public string[] vertex;

        //保存邊信息
        public int[,] edges;

        //深搜和廣搜的遍歷標志
        public bool[] isTrav;

        //頂點數(shù)量
        public int vertexNum;

        //邊數(shù)量
        public int edgeNum;

        //圖類型
        public int graphType;

        /// <summary>
/// 存儲容量的初始化
/// </summary>
/// <param name="vertexNum"></param>
/// <param name="edgeNum"></param>
/// <param name="graphType"></param>
        public MatrixGraph(int vertexNum, int edgeNum, int graphType)
        {
            this.vertexNum = vertexNum;
            this.edgeNum = edgeNum;
            this.graphType = graphType;

            vertex = new string[vertexNum];
            edges = new int[vertexNum, vertexNum];
            isTrav = new bool[vertexNum];
        }

    }
    #endregion

<1> 創(chuàng)建圖很簡單,讓用戶輸入一些“邊,點,權(quán)值"來構(gòu)建一下圖

復制代碼 代碼如下:

#region 圖的創(chuàng)建
        /// <summary>
/// 圖的創(chuàng)建
/// </summary>
/// <param name="g"></param>
        public MatrixGraph CreateMatrixGraph()
        {
            Console.WriteLine("請輸入創(chuàng)建圖的頂點個數(shù),邊個數(shù),是否為無向圖(0,1來表示),已逗號隔開。");

            var initData = Console.ReadLine().Split(',').Select(i => int.Parse(i)).ToList();

            MatrixGraph graph = new MatrixGraph(initData[0], initData[1], initData[2]);

            Console.WriteLine("請輸入各頂點信息:");

            for (int i = 0; i < graph.vertexNum; i++)
            {
                Console.Write("/n第" + (i + 1) + "個頂點為:");

                var single = Console.ReadLine();

                //頂點信息加入集合中
                graph.vertex[i] = single;
            }

            Console.WriteLine("/n請輸入構(gòu)成兩個頂點的邊和權(quán)值,以逗號隔開。/n");

            for (int i = 0; i < graph.edgeNum; i++)
            {
                Console.Write("第" + (i + 1) + "條邊:/t");

                initData = Console.ReadLine().Split(',').Select(j => int.Parse(j)).ToList();

                int start = initData[0];
                int end = initData[1];
                int weight = initData[2];

                //給矩陣指定坐標位置賦值
                graph.edges[start - 1, end - 1] = weight;

                //如果是無向圖,則數(shù)據(jù)呈“二,四”象限對稱
                if (graph.graphType == 1)
                {
                    graph.edges[end - 1, start - 1] = weight;
                }
            }

            return graph;
        }
        #endregion

<2>廣度優(yōu)先

      針對下面的“圖型結(jié)構(gòu)”,我們?nèi)绾螐V度優(yōu)先呢?其實我們只要深刻理解"廣搜“給我們定義的條條框框就行了。 為了避免同一個頂點在遍歷時被多

次訪問,可以將”頂點的下標”存放在sTrav[]的bool數(shù)組,用來標識是否已經(jīng)訪問過該節(jié)點。 

    第一步:首先我們從isTrav數(shù)組中選出一個未被訪問的節(jié)點,如V1。

    第二步:訪問V1的鄰接點V2,V3,V5,并將這三個節(jié)點標記為true。

    第三步:第二步結(jié)束后,我們開始訪問V2的鄰接點V1,V3,但是他們都是被訪問過的。

    第四步:我們從第二步結(jié)束的V3出發(fā)訪問他的鄰接點V2,V1,V5,V4,還好V4是未被訪問的,此時標記一下。

    第五步:我們訪問V5的鄰接點V1,V3,V4,不過都是已經(jīng)訪問過的。

    第六步:有的圖中通過一個頂點的“廣度優(yōu)先”不能遍歷所有的頂點,此時我們重復(1-5)的步驟就可以最終完成廣度優(yōu)先遍歷。



復制代碼 代碼如下:

#region 廣度優(yōu)先
        /// <summary>
/// 廣度優(yōu)先
/// </summary>
/// <param name="graph"></param>
        public void BFSTraverse(MatrixGraph graph)
        {
            //訪問標記默認初始化
            for (int i = 0; i < graph.vertexNum; i++)
            {
                graph.isTrav[i] = false;
            }

            //遍歷每個頂點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                //廣度遍歷未訪問過的頂點
                if (!graph.isTrav[i])
                {
                    BFSM(ref graph, i);
                }
            }
        }

        /// <summary>
/// 廣度遍歷具體算法
/// </summary>
/// <param name="graph"></param>
        public void BFSM(ref MatrixGraph graph, int vertex)
        {
            //這里就用系統(tǒng)的隊列
            Queue<int> queue = new Queue<int>();

            //先把頂點入隊
            queue.Enqueue(vertex);

            //標記此頂點已經(jīng)被訪問
            graph.isTrav[vertex] = true;

            //輸出頂點
            Console.Write(" ->" + graph.vertex[vertex]);

            //廣度遍歷頂點的鄰接點
            while (queue.Count != 0)
            {
                var temp = queue.Dequeue();

                //遍歷矩陣的橫坐標
                for (int i = 0; i < graph.vertexNum; i++)
                {
                    if (!graph.isTrav[i] && graph.edges[temp, i] != 0)
                    {
                        graph.isTrav[i] = true;

                        queue.Enqueue(i);

                        //輸出未被訪問的頂點
                        Console.Write(" ->" + graph.vertex[i]);
                    }
                }
            }
        }
        #endregion

<3> 深度優(yōu)先

        同樣是這個圖,大家看看如何實現(xiàn)深度優(yōu)先,深度優(yōu)先就像鐵骨錚錚的好漢,遵循“能進則進,不進則退”的原則。

        第一步:同樣也是從isTrav數(shù)組中選出一個未被訪問的節(jié)點,如V1。

        第二步:然后一直訪問V1的鄰接點,一直到走頭無路的時候“回溯”,路線為V1,V2,V3,V4,V5,到V5的時候訪問鄰接點V1,發(fā)現(xiàn)V1是訪問過的,

                   此時一直回溯的訪問直到V1。

        第三步: 同樣有的圖中通過一個頂點的“深度優(yōu)先”不能遍歷所有的頂點,此時我們重復(1-2)的步驟就可以最終完成深度優(yōu)先遍歷。

              

復制代碼 代碼如下:

#region 深度優(yōu)先
        /// <summary>
/// 深度優(yōu)先
/// </summary>
/// <param name="graph"></param>
        public void DFSTraverse(MatrixGraph graph)
        {
            //訪問標記默認初始化
            for (int i = 0; i < graph.vertexNum; i++)
            {
                graph.isTrav[i] = false;
            }

            //遍歷每個頂點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                //廣度遍歷未訪問過的頂點
                if (!graph.isTrav[i])
                {
                    DFSM(ref graph, i);
                }
            }
        }

        #region 深度遞歸的具體算法
        /// <summary>
/// 深度遞歸的具體算法
/// </summary>
/// <param name="graph"></param>
/// <param name="vertex"></param>
        public void DFSM(ref MatrixGraph graph, int vertex)
        {
            Console.Write("->" + graph.vertex[vertex]);

            //標記為已訪問
            graph.isTrav[vertex] = true;

            //要遍歷的六個點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                if (graph.isTrav[i] == false && graph.edges[vertex, i] != 0)
                {
                    //深度遞歸
                    DFSM(ref graph, i);
                }
            }
        }
        #endregion
        #endregion

最后上一下總的代碼

復制代碼 代碼如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace MatrixGraph
{
    public class Program
    {
        static void Main(string[] args)
        {
            MatrixGraphManager manager = new MatrixGraphManager();

            //創(chuàng)建圖
            MatrixGraph graph = manager.CreateMatrixGraph();

            manager.OutMatrix(graph);

            Console.Write("廣度遞歸:/t");

            manager.BFSTraverse(graph);

            Console.Write("/n深度遞歸:/t");

            manager.DFSTraverse(graph);

            Console.ReadLine();

        }
    }

    #region 鄰接矩陣的結(jié)構(gòu)圖
    /// <summary>
/// 鄰接矩陣的結(jié)構(gòu)圖
/// </summary>
    public class MatrixGraph
    {
        //保存頂點信息
        public string[] vertex;

        //保存邊信息
        public int[,] edges;

        //深搜和廣搜的遍歷標志
        public bool[] isTrav;

        //頂點數(shù)量
        public int vertexNum;

        //邊數(shù)量
        public int edgeNum;

        //圖類型
        public int graphType;

        /// <summary>
/// 存儲容量的初始化
/// </summary>
/// <param name="vertexNum"></param>
/// <param name="edgeNum"></param>
/// <param name="graphType"></param>
        public MatrixGraph(int vertexNum, int edgeNum, int graphType)
        {
            this.vertexNum = vertexNum;
            this.edgeNum = edgeNum;
            this.graphType = graphType;

            vertex = new string[vertexNum];
            edges = new int[vertexNum, vertexNum];
            isTrav = new bool[vertexNum];
        }

    }
    #endregion

    /// <summary>
/// 圖的操作類
/// </summary>
    public class MatrixGraphManager
    {
        #region 圖的創(chuàng)建
        /// <summary>
/// 圖的創(chuàng)建
/// </summary>
/// <param name="g"></param>
        public MatrixGraph CreateMatrixGraph()
        {
            Console.WriteLine("請輸入創(chuàng)建圖的頂點個數(shù),邊個數(shù),是否為無向圖(0,1來表示),已逗號隔開。");

            var initData = Console.ReadLine().Split(',').Select(i => int.Parse(i)).ToList();

            MatrixGraph graph = new MatrixGraph(initData[0], initData[1], initData[2]);

            Console.WriteLine("請輸入各頂點信息:");

            for (int i = 0; i < graph.vertexNum; i++)
            {
                Console.Write("/n第" + (i + 1) + "個頂點為:");

                var single = Console.ReadLine();

                //頂點信息加入集合中
                graph.vertex[i] = single;
            }

            Console.WriteLine("/n請輸入構(gòu)成兩個頂點的邊和權(quán)值,以逗號隔開。/n");

            for (int i = 0; i < graph.edgeNum; i++)
            {
                Console.Write("第" + (i + 1) + "條邊:/t");

                initData = Console.ReadLine().Split(',').Select(j => int.Parse(j)).ToList();

                int start = initData[0];
                int end = initData[1];
                int weight = initData[2];

                //給矩陣指定坐標位置賦值
                graph.edges[start - 1, end - 1] = weight;

                //如果是無向圖,則數(shù)據(jù)呈“二,四”象限對稱
                if (graph.graphType == 1)
                {
                    graph.edges[end - 1, start - 1] = weight;
                }
            }

            return graph;
        }
        #endregion

        #region 輸出矩陣數(shù)據(jù)
        /// <summary>
/// 輸出矩陣數(shù)據(jù)
/// </summary>
/// <param name="graph"></param>
        public void OutMatrix(MatrixGraph graph)
        {
            for (int i = 0; i < graph.vertexNum; i++)
            {
                for (int j = 0; j < graph.vertexNum; j++)
                {
                    Console.Write(graph.edges[i, j] + "/t");
                }
                //換行
                Console.WriteLine();
            }
        }
        #endregion

        #region 廣度優(yōu)先
        /// <summary>
/// 廣度優(yōu)先
/// </summary>
/// <param name="graph"></param>
        public void BFSTraverse(MatrixGraph graph)
        {
            //訪問標記默認初始化
            for (int i = 0; i < graph.vertexNum; i++)
            {
                graph.isTrav[i] = false;
            }

            //遍歷每個頂點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                //廣度遍歷未訪問過的頂點
                if (!graph.isTrav[i])
                {
                    BFSM(ref graph, i);
                }
            }
        }

        /// <summary>
/// 廣度遍歷具體算法
/// </summary>
/// <param name="graph"></param>
        public void BFSM(ref MatrixGraph graph, int vertex)
        {
            //這里就用系統(tǒng)的隊列
            Queue<int> queue = new Queue<int>();

            //先把頂點入隊
            queue.Enqueue(vertex);

            //標記此頂點已經(jīng)被訪問
            graph.isTrav[vertex] = true;

            //輸出頂點
            Console.Write(" ->" + graph.vertex[vertex]);

            //廣度遍歷頂點的鄰接點
            while (queue.Count != 0)
            {
                var temp = queue.Dequeue();

                //遍歷矩陣的橫坐標
                for (int i = 0; i < graph.vertexNum; i++)
                {
                    if (!graph.isTrav[i] && graph.edges[temp, i] != 0)
                    {
                        graph.isTrav[i] = true;

                        queue.Enqueue(i);

                        //輸出未被訪問的頂點
                        Console.Write(" ->" + graph.vertex[i]);
                    }
                }
            }
        }
        #endregion

        #region 深度優(yōu)先
        /// <summary>
/// 深度優(yōu)先
/// </summary>
/// <param name="graph"></param>
        public void DFSTraverse(MatrixGraph graph)
        {
            //訪問標記默認初始化
            for (int i = 0; i < graph.vertexNum; i++)
            {
                graph.isTrav[i] = false;
            }

            //遍歷每個頂點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                //廣度遍歷未訪問過的頂點
                if (!graph.isTrav[i])
                {
                    DFSM(ref graph, i);
                }
            }
        }

        #region 深度遞歸的具體算法
        /// <summary>
/// 深度遞歸的具體算法
/// </summary>
/// <param name="graph"></param>
/// <param name="vertex"></param>
        public void DFSM(ref MatrixGraph graph, int vertex)
        {
            Console.Write("->" + graph.vertex[vertex]);

            //標記為已訪問
            graph.isTrav[vertex] = true;

            //要遍歷的六個點
            for (int i = 0; i < graph.vertexNum; i++)
            {
                if (graph.isTrav[i] == false && graph.edges[vertex, i] != 0)
                {
                    //深度遞歸
                    DFSM(ref graph, i);
                }
            }
        }
        #endregion
        #endregion

    }
}

代碼中我們構(gòu)建了如下的“圖”。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
国产网友自拍电影在线| 91九色在线看| 九九免费视频| 久久五月精品| 国产剧情在线一区| 国产色婷婷在线| 国产成+人+亚洲+欧美+综合| 91精选福利| 好男人免费精品视频| 国产精品日日爱| 国产麻豆精品高清在线播放| 国产乱子视频| 精品视频vs精品视频| www.国产精品.com| 五月婷婷在线视频| 亚洲欧美精品日韩欧美| 黄色av网站在线| 99爱视频在线观看| 国产一二区在线| 交视频在线观看国产| 18av在线播放| 国产乱xxⅹxx国语对白| 国产91足控脚交在线观看| 麻豆视频在线观看免费网站| 中文字幕一区二区三区免费视频| 国产美女性感在线观看懂色av| 国产精彩视频在线观看免费蜜芽| 日本aⅴ写真网站免费| 精品国产免费观看一区| 午夜视频99| 黄色网页网址在线免费| 国产成人综合美国十次| 麻豆视频在线观看免费网站| 91午夜在线| 最近免费中文字幕大全免费第三页| 97高清视频| 国产精品秘入口| 国产一级激情| 天堂在线一二区| 久久国产热视频| 69国产精品| 在线中文av| 亚洲夜夜综合| av小说在线| 国产xxx在线| 国产成人精品18| 国产成人福利| 夜夜爽视频导航| 亚洲夜夜综合| 日本成人在线播放| 秋霞av在线| 青草青在线视频| 黄网站在线观看高清免费| av在线免费观看网| 四虎a级欧美在线观看| 中文字幕2020第一页| 中文岛国精品亚洲一区| 在线观看电影av| 青青草原国产在线| 国产精品探花在线| 国产污污在线观看| 99热免费在线| 国产在线传媒| 国产精品666| 欧美精品久久久久久久小说| 狠狠干天天爱| 国产精品国精产品一二| 亚洲欧美久久婷婷爱综合一区天堂| 亚洲欧美精品日韩欧美| 欧美在线中文| 成人亚洲一区二区三区| 国产视频在线播放| a视频免费看| 国产馆av播放| 青青青青在线| 尤物视频在线观看视频| 天堂在线免费视频| 性国产高清在线观看| 国产免费麻豆视频| 在线色视频观看| 丁香婷婷在线观看| 91午夜在线| 国产精品综合久久久久| 在线天堂av| 国产精品美女一区二区视频| 欧美高清视频| 亚洲国产成人综合| 国产9色视频| 国产一级视频| 国产免费福利| 亚洲成人在线播放| 99热99re6国产在线播放| 88av在线| 国产在线精品一区二区不卡| 91嫩草在线播放| 国产乱精品一区二区三区| 亚洲an天堂an在线观看| 精品女厕厕露p撒尿| 国产欧美日韩第一页| 国产在线传媒| 老师我好爽再深一点的视频| 亚洲欧美日韩成人网| 中文av资源在线| 99热免费在线| 亚洲第一页在线播放| 99reav| 国产区在线看| 国产一区二区在线|播放| 激情亚洲综合网| 狠狠干天天爱| 国产一区二区影视| 国产精品第八页| 97一区二区三区| 四虎成人免费观看在线网址| 小说区乱图片区| 青青青青在线| 国产三级在线播放| 一本大道久久a久久精品| 欧美色欧美亚洲另类二区精品| 国产在线观看色| 国产三区视频在线观看| 伊人永久在线| 六月天色婷婷| 国产视频中文字幕在线观看| 精灵使的剑舞无删减版在线观看| 任你操在线观看| 国产深夜福利| 欧美在线中文| 97影院理论午夜| 国产卡二和卡三的视频| 国产精品久久久久久久牛牛| 在线国产一级| 国产美女被遭强高潮免费网站| 国产黄在线观看| 麻豆网站在线免费观看| 国产对白国语对白| 狠狠色丁香婷婷| gogo在线观看| 国产导航在线| 高清视频一区二区三区四区| 国产黄色在线网站| 国产美女福利在线| 亚洲第一页在线播放| 国产精品粉嫩av| 国产免费永久在线观看| 99re在线视频播放| 国产精品久久人| 国产免费福利| 欧美日韩亚洲国内综合网| 国产黄色高清在线| 五月婷婷视频在线观看| 久久99国产视频| jlzzjlzz欧美| sese在线视频| av网站大全在线观看| 2019年中文字幕| 国产精品久久久久永久免费看| 国产伦精品一区二区三区高清版禁| 日本a级黄色| 欧美成人久久电影香蕉| 四虎精品成人a在线观看| 91涩漫在线观看c| 91网页在线观看| 精品无人乱码| 国产盗摄一区二区| 国产va在线| 精品51国产黑色丝袜高跟鞋| 国产馆av播放| 中文在线观看视频| 亚洲私人影吧| 欧美性猛交xxxx免费看蜜桃| 日本在线观看网站| 男人天堂99| 最近中文字幕mv免费高清在线| 久久精品无码一区二区日韩av | 超碰91在线| 精品国产高清a毛片无毒不卡| 国产欧美日韩专区| 国产一级片网站| 91最新在线| www.三区| 国产网站观看9久| 精品国产白色丝袜高跟鞋| 精品亚洲成a人片在线观看| 96精品视频| 性国产高清在线观看| 久热国产视频| 国产女王在线**视频 | 精品一区二区91| 在线中文字幕观看| www.久草.com| 国产网站免费观看| 精品黄色免费中文电影在线播放| jlzzjlzz欧美| 国产午夜视频在线观看| eeuss在线观看| 国产系列在线观看| wwwww亚洲| 精品麻豆视频| 中文字幕欧美日韩在线不卡|