1. package com.atguigu.graph;
    2. import java.util.ArrayList;
    3. import java.util.Arrays;
    4. import java.util.LinkedList;
    5. /**
    6. * ClassName: <br/>
    7. * Description: <br/>
    8. * Date: 2021-03-04 11:36 <br/>
    9. * @project data_algorithm
    10. * @package com.atguigu.graph
    11. */
    12. public class Graph {
    13. private ArrayList<String> vertexList; //存储顶点集合
    14. private int[][] edges; //存储图对应的邻结矩阵
    15. private int numOfEdges; //表示边的数目
    16. //定义给数组boolean[], 记录某个结点是否被访问
    17. private boolean[] isVisited;
    18. public static void main(String[] args) {
    19. //测试一把图是否创建ok
    20. int n = 8; //结点的个数
    21. //String Vertexs[] = {"A", "B", "C", "D", "E"};
    22. String Vertexs[] = {"1", "2", "3", "4", "5", "6", "7", "8"};
    23. //创建图对象
    24. Graph graph = new Graph(n);
    25. //循环的添加顶点
    26. for(String vertex: Vertexs) {
    27. graph.insertVertex(vertex);
    28. }
    29. //添加边
    30. //A-B A-C B-C B-D B-E
    31. // graph.insertEdge(0, 1, 1); // A-B
    32. // graph.insertEdge(0, 2, 1); //
    33. // graph.insertEdge(1, 2, 1); //
    34. // graph.insertEdge(1, 3, 1); //
    35. // graph.insertEdge(1, 4, 1); //
    36. //更新边的关系
    37. graph.insertEdge(0, 1, 1);
    38. graph.insertEdge(0, 2, 1);
    39. graph.insertEdge(1, 3, 1);
    40. graph.insertEdge(1, 4, 1);
    41. graph.insertEdge(3, 7, 1);
    42. graph.insertEdge(4, 7, 1);
    43. graph.insertEdge(2, 5, 1);
    44. graph.insertEdge(2, 6, 1);
    45. graph.insertEdge(5, 6, 1);
    46. //显示一把邻结矩阵
    47. graph.showGraph();
    48. //测试一把,我们的dfs遍历是否ok
    49. System.out.println("深度遍历");
    50. graph.dfs(); // A->B->C->D->E [1->2->4->8->5->3->6->7]
    51. // System.out.println();
    52. System.out.println("广度优先!");
    53. graph.bfs(); // A->B->C->D-E [1->2->3->4->5->6->7->8]
    54. }
    55. //构造器
    56. public Graph(int n) {
    57. //初始化矩阵和vertexList
    58. edges = new int[n][n];
    59. vertexList = new ArrayList<String>(n);
    60. numOfEdges = 0;
    61. }
    62. //得到第一个邻接结点的下标 w
    63. /**
    64. *
    65. * @param index
    66. * @return 如果存在就返回对应的下标,否则返回-1
    67. */
    68. public int getFirstNeighbor(int index) {
    69. for(int j = 0; j < vertexList.size(); j++) {
    70. if(edges[index][j] > 0) {
    71. return j;
    72. }
    73. }
    74. return -1;
    75. }
    76. //根据前一个邻接结点的下标来获取下一个邻接结点
    77. public int getNextNeighbor(int v1, int v2) {
    78. for(int j = v2 + 1; j < vertexList.size(); j++) {
    79. if(edges[v1][j] > 0) {
    80. return j;
    81. }
    82. }
    83. return -1;
    84. }
    85. //深度优先遍历算法
    86. //i 第一次就是 0
    87. private void dfs(boolean[] isVisited, int i) {
    88. //首先我们访问该结点,输出
    89. System.out.print(getValueByIndex(i) + "->");
    90. //将结点设置为已经访问
    91. isVisited[i] = true;
    92. //查找结点i的第一个邻接结点w
    93. int w = getFirstNeighbor(i);
    94. while(w != -1) {//说明有
    95. if(!isVisited[w]) {
    96. dfs(isVisited, w);
    97. }
    98. //如果w结点已经被访问过
    99. w = getNextNeighbor(i, w);
    100. }
    101. }
    102. //对dfs 进行一个重载, 遍历我们所有的结点,并进行 dfs
    103. public void dfs() {
    104. isVisited = new boolean[vertexList.size()];
    105. //遍历所有的结点,进行dfs[回溯]
    106. for(int i = 0; i < getNumOfVertex(); i++) {
    107. if(!isVisited[i]) {
    108. dfs(isVisited, i);
    109. }
    110. }
    111. }
    112. //对一个结点进行广度优先遍历的方法
    113. private void bfs(boolean[] isVisited, int i) {
    114. int u ; // 表示队列的头结点对应下标
    115. int w ; // 邻接结点w
    116. //队列,记录结点访问的顺序
    117. LinkedList queue = new LinkedList();
    118. //访问结点,输出结点信息
    119. System.out.print(getValueByIndex(i) + "=>");
    120. //标记为已访问
    121. isVisited[i] = true;
    122. //将结点加入队列
    123. queue.addLast(i);
    124. while( !queue.isEmpty()) {
    125. //取出队列的头结点下标
    126. u = (Integer)queue.removeFirst();
    127. //得到第一个邻接结点的下标 w
    128. w = getFirstNeighbor(u);
    129. while(w != -1) {//找到
    130. //是否访问过
    131. if(!isVisited[w]) {
    132. System.out.print(getValueByIndex(w) + "=>");
    133. //标记已经访问
    134. isVisited[w] = true;
    135. //入队
    136. queue.addLast(w);
    137. }
    138. //以u为前驱点,找w后面的下一个邻结点
    139. w = getNextNeighbor(u, w); //体现出我们的广度优先
    140. }
    141. }
    142. }
    143. //遍历所有的结点,都进行广度优先搜索
    144. public void bfs() {
    145. isVisited = new boolean[vertexList.size()];
    146. for(int i = 0; i < getNumOfVertex(); i++) {
    147. if(!isVisited[i]) {
    148. bfs(isVisited, i);
    149. }
    150. }
    151. }
    152. //图中常用的方法
    153. //返回结点的个数
    154. public int getNumOfVertex() {
    155. return vertexList.size();
    156. }
    157. //显示图对应的矩阵
    158. public void showGraph() {
    159. for(int[] link : edges) {
    160. System.err.println(Arrays.toString(link));
    161. }
    162. }
    163. //得到边的数目
    164. public int getNumOfEdges() {
    165. return numOfEdges;
    166. }
    167. //返回结点i(下标)对应的数据 0->"A" 1->"B" 2->"C"
    168. public String getValueByIndex(int i) {
    169. return vertexList.get(i);
    170. }
    171. //返回v1和v2的权值
    172. public int getWeight(int v1, int v2) {
    173. return edges[v1][v2];
    174. }
    175. //插入结点
    176. public void insertVertex(String vertex) {
    177. vertexList.add(vertex);
    178. }
    179. //添加边
    180. /**
    181. *
    182. * @param v1 表示点的下标即使第几个顶点 "A"-"B" "A"->0 "B"->1
    183. * @param v2 第二个顶点对应的下标
    184. * @param weight 表示
    185. */
    186. public void insertEdge(int v1, int v2, int weight) {
    187. edges[v1][v2] = weight;
    188. edges[v2][v1] = weight;
    189. numOfEdges++;
    190. }
    191. }

    输出

    1. [0, 1, 1, 0, 0, 0, 0, 0]
    2. [1, 0, 0, 1, 1, 0, 0, 0]
    3. [1, 0, 0, 0, 0, 1, 1, 0]
    4. [0, 1, 0, 0, 0, 0, 0, 1]
    5. [0, 1, 0, 0, 0, 0, 0, 1]
    6. [0, 0, 1, 0, 0, 0, 1, 0]
    7. [0, 0, 1, 0, 0, 1, 0, 0]
    8. [0, 0, 0, 1, 1, 0, 0, 0]
    9. 深度遍历
    10. 1->2->4->8->5->3->6->7->广度优先!
    11. 1=>2=>3=>4=>5=>6=>7=>8=>
    12. Process finished with exit code 0