所谓图的遍历,即是对结点的访问。一个图有那么多个结点,如何遍历这些结点,需要特定策略,一般有两种访问策略: (1)深度优先遍历 (2)广度优先遍历
图的深度优先搜索(Depth First Search) 。
1) 要求:对下图进行深度优先搜索, 从A 开始遍历.
package graph;
import java.util.ArrayList;
import java.util.Arrays;
public class Graph {
private ArrayList vertexList; // 存储节点的集合
private int[][] edges; // 存储对应的邻接矩阵
private int numberOfEdges; // 表示边的数目
//定义给数组boolean[],记录某个结点是否被访问
private boolean[] isVisited;
public static void main(String[] args) {
int n = 5; // 节点的个数
String[] vertexValue = {"A", "B", "C", "D", "E"};
// 创建图对象
Graph graph = new Graph(n);
// 循环添加节点数据
for (String vertex : vertexValue) {
graph.insertVertex(vertex);
}
// 添加边 A-B A-C B-C B-D B-E
graph.insertEdge(0, 1, 1);
graph.insertEdge(0, 2, 1);
graph.insertEdge(1, 2, 1);
graph.insertEdge(1, 3, 1);
graph.insertEdge(1, 4, 1);
graph.showGraph();
//测试一个,我们的dfs是否ok
System.out.println("深度优先遍历");
graph.dfs();
}
public Graph (int n) {
// 初始化矩阵 和 arrayList
edges = new int[n][n];
vertexList = new ArrayList<>(n);
numberOfEdges = 0;
isVisited=new boolean[5];
}
/*
* @Param index
* @return 如果存在就返回对应下标 否则返回-1
* */
//得到第一个邻接结点的下标w
public int getFirstNeighbor(int index){
for(int j=0;j0){
return j;
}
}
return -1;
}
//根据前一个邻接结点的下标来获取下一个邻接结点
public int getNextNeighbor(int v1,int v2){
for (int j=v2+1;j0){
return j;
}
}return -1;
}
//深度优先遍历算法
//i 第一次就是0
private void dfs(boolean[] isVisited,int i) {
//首先我们访问该结点 输出
System.out.print(getValueByIndex(i) + "->");
//将结点设置为已访问
isVisited[i] = true;
//查找结点i的第一个邻接结点w
int w = getFirstNeighbor(i);
while (w != -1) {//说明有
if (!isVisited[w]){
dfs(isVisited,w);
}
//如果结点已经被访问过
w=getNextNeighbor(i,w);
}
}
//对dfs进行重载,遍历我们所有的结点,并进行dfs
public void dfs(){
//遍历所有的结点,进行dfs[回溯]
for (int i=0;i"A" 1->"B"
public String getValueByIndex(int i) {
return vertexList.get(i);
}
// 4. 返回v1 和 v2 边的权值
public int insertVertex(int v1, int v2) {
return edges[v1][v2];
}
// 5. 显示图对应的矩阵
public void showGraph() {
System.out.println("遍历矩阵 ");
for (int[] link : edges) {
System.out.println(Arrays.toString(link));
}
}
// 插入节点
public void insertVertex(String vertex) {
vertexList.add(vertex);
}
// 添加边
/**
*
* @param v1 第一个顶点的下标 第几个顶点
* @param v2 第二个顶点的下标
* @param weight 边的权值
*/
public void insertEdge(int v1, int v2, int weight) {
edges[v1][v2] = weight;
edges[v2][v1] = weight;
numberOfEdges++;
}
}
图的广度优先搜索(Broad First Search) 。
类似于一个分层搜索的过程,广度优先遍历需要使用一个队列以保持访问过的结点的顺序,以便按这个顺序来访问这些结点的邻接结点!
6.1 若结点w尚未被访问,则访问结点w并标记为已访问
6.2 结点w入队列
6.3 查找结点u的继w邻接结点后的下一个邻接结点w,转到步骤6。
增加以下两个方法
//对一个结点进行广度优先遍历的方法
public void bfs(boolean[] isVisited,int i){
int u;//表示队列的头结点对应下标
int w; //邻接结点w
//队列 记录结点访问的顺序
LinkedList queue = new LinkedList<>();
System.out.print(getValueByIndex(i)+"=>");
//标记为已访问
isVisited[i]=true;
//将结点加入队列
queue.addLast(i);
while (!queue.isEmpty()){
//取出队列的头结点的下标
u=(Integer)queue.removeFirst();
//得到第一个邻接点的下标w
w=getFirstNeighbor(u);
while (w!=-1){
//找到
//是否访问过
if (!isVisited[w]){
System.out.print(getValueByIndex(w)+"=>");
//标记已经访问过
isVisited[w]=true;
//入队列
queue.addLast(w);
}
//以u为前驱点,找w后面的下一个邻接点
w=getNextNeighbor(u,w);//体现出我们的广度优先
}
}
}
//遍历所有的结点,都进行广度优先搜索
public void bfs(){
isVisited=new boolean[5];
for (int i=0;i
package graph;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
public class Graph {
private ArrayList vertexList; // 存储节点的集合
private int[][] edges; // 存储对应的邻接矩阵
private int numberOfEdges; // 表示边的数目
//定义给数组boolean[],记录某个结点是否被访问
private boolean[] isVisited;
public static void main(String[] args) {
int n = 5; // 节点的个数
String[] vertexValue = {"A", "B", "C", "D", "E"};
// 创建图对象
Graph graph = new Graph(n);
// 循环添加节点数据
for (String vertex : vertexValue) {
graph.insertVertex(vertex);
}
// 添加边 A-B A-C B-C B-D B-E
graph.insertEdge(0, 1, 1);
graph.insertEdge(0, 2, 1);
graph.insertEdge(1, 2, 1);
graph.insertEdge(1, 3, 1);
graph.insertEdge(1, 4, 1);
graph.showGraph();
//测试一个,我们的dfs是否ok
System.out.println("深度优先遍历");
graph.dfs();
System.out.println("广度优先遍历");
graph.bfs();
}
public Graph (int n) {
// 初始化矩阵 和 arrayList
edges = new int[n][n];
vertexList = new ArrayList<>(n);
numberOfEdges = 0;
isVisited=new boolean[5];
}
/*
* @Param index
* @return 如果存在就返回对应下标 否则返回-1
* */
//得到第一个邻接结点的下标
public int getFirstNeighbor(int index){
for(int j=0;j0){
return j;
}
}
return -1;
}
//根据前一个邻接结点的下标来获取下一个邻接结点
public int getNextNeighbor(int v1,int v2){
for (int j=v2+1;j0){
return j;
}
}return -1;
}
//深度优先遍历算法
//i 第一次就是0
private void dfs(boolean[] isVisited,int i) {//a
//首先我们访问该结点 输出
System.out.print(getValueByIndex(i) + "->");
//将结点设置为已访问
isVisited[i] = true;
//查找结点i的第一个邻接结点w
int w = getFirstNeighbor(i);
while (w != -1) {//说明有
if (!isVisited[w]){
dfs(isVisited,w);//b-》c-》
}
//如果结点已经被访问过
w=getNextNeighbor(i,w);
}
}
//对dfs进行重载,遍历我们所有的结点,并进行dfs
public void dfs(){
isVisited=new boolean[5];
//遍历所有的结点,进行dfs[回溯]
for (int i=0;i"A" 1->"B"
public String getValueByIndex(int i) {
return vertexList.get(i);
}
// 4. 返回v1 和 v2 边的权值
public int insertVertex(int v1, int v2) {
return edges[v1][v2];
}
// 5. 显示图对应的矩阵
public void showGraph() {
System.out.println("遍历矩阵 ");
for (int[] link : edges) {
System.out.println(Arrays.toString(link));
}
}
// 插入节点
public void insertVertex(String vertex) {
vertexList.add(vertex);
}
// 添加边
/**
*
* @param v1 第一个顶点的下标 第几个顶点
* @param v2 第二个顶点的下标
* @param weight 边的权值
*/
public void insertEdge(int v1, int v2, int weight) {
edges[v1][v2] = weight;
edges[v2][v1] = weight;
numberOfEdges++;
}
//对一个结点进行广度优先遍历的方法
public void bfs(boolean[] isVisited,int i){
int u;//表示队列的头结点对应下标
int w; //邻接结点w
//队列 记录结点访问的顺序
LinkedList queue = new LinkedList<>();
System.out.print(getValueByIndex(i)+"=>");
//标记为已访问
isVisited[i]=true;
//将结点加入队列
queue.addLast(i);
while (!queue.isEmpty()){
//取出队列的头结点的下标
u=(Integer)queue.removeFirst();
//得到第一个邻接点的下标w
w=getFirstNeighbor(u);
while (w!=-1){
//找到
//是否访问过
if (!isVisited[w]){
System.out.print(getValueByIndex(w)+"=>");
//标记已经访问过
isVisited[w]=true;
//入队列
queue.addLast(w);
}
//以u为前驱点,找w后面的下一个邻接点
w=getNextNeighbor(u,w);//体现出我们的广度优先
}
}
}
//遍历所有的结点,都进行广度优先搜索
public void bfs(){
isVisited=new boolean[5];
for (int i=0;i
由于此测试用例深度或者广度遍历输出的都是一样的结果,因此增加一个测试用例:
package graph;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
public class Graph {
private ArrayList vertexList; // 存储节点的集合
private int[][] edges; // 存储对应的邻接矩阵
private int numberOfEdges; // 表示边的数目
//定义给数组boolean[],记录某个结点是否被访问
private boolean[] isVisited;
public static void main(String[] args) {
int n = 8; // 节点的个数
// String[] vertexValue = {"A", "B", "C", "D", "E"};
String[] vertexValue = {"1", "2", "3", "4", "5","6","7","8"};
// 创建图对象
Graph graph = new Graph(n);
// 循环添加节点数据
for (String vertex : vertexValue) {
graph.insertVertex(vertex);
}
// 添加边 A-B A-C B-C B-D B-E
/* graph.insertEdge(0, 1, 1);
graph.insertEdge(0, 2, 1);
graph.insertEdge(1, 2, 1);
graph.insertEdge(1, 3, 1);
graph.insertEdge(1, 4, 1);*/
//更新边的关系
graph.insertEdge(0, 1, 1);
graph.insertEdge(0, 2, 1);
graph.insertEdge(1, 3, 1);
graph.insertEdge(1, 4, 1);
graph.insertEdge(3, 7, 1);
graph.insertEdge(4, 7, 1);
graph.insertEdge(2, 5, 1);
graph.insertEdge(2, 6, 1);
graph.insertEdge(5, 6, 1);
graph.showGraph();
//测试一个,我们的dfs是否ok
System.out.println("深度优先遍历");
graph.dfs();
System.out.println("广度优先遍历");
graph.bfs();
}
public Graph (int n) {
// 初始化矩阵 和 arrayList
edges = new int[n][n];
vertexList = new ArrayList<>(n);
numberOfEdges = 0;
isVisited=new boolean[5];
}
/*
* @Param index
* @return 如果存在就返回对应下标 否则返回-1
* */
//得到第一个邻接结点的下标
public int getFirstNeighbor(int index){
for(int j=0;j0){
return j;
}
}
return -1;
}
//根据前一个邻接结点的下标来获取下一个邻接结点
public int getNextNeighbor(int v1,int v2){
for (int j=v2+1;j0){
return j;
}
}return -1;
}
//深度优先遍历算法
//i 第一次就是0
private void dfs(boolean[] isVisited,int i) {//a
//首先我们访问该结点 输出
System.out.print(getValueByIndex(i) + "->");
//将结点设置为已访问
isVisited[i] = true;
//查找结点i的第一个邻接结点w
int w = getFirstNeighbor(i);
while (w != -1) {//说明有
if (!isVisited[w]){
dfs(isVisited,w);//b-》c-》
}
//如果结点已经被访问过
w=getNextNeighbor(i,w);
}
}
//对dfs进行重载,遍历我们所有的结点,并进行dfs
public void dfs(){
isVisited=new boolean[8];
//遍历所有的结点,进行dfs[回溯]
for (int i=0;i"A" 1->"B"
public String getValueByIndex(int i) {
return vertexList.get(i);
}
// 4. 返回v1 和 v2 边的权值
public int insertVertex(int v1, int v2) {
return edges[v1][v2];
}
// 5. 显示图对应的矩阵
public void showGraph() {
System.out.println("遍历矩阵 ");
for (int[] link : edges) {
System.out.println(Arrays.toString(link));
}
}
// 插入节点
public void insertVertex(String vertex) {
vertexList.add(vertex);
}
// 添加边
/**
*
* @param v1 第一个顶点的下标 第几个顶点
* @param v2 第二个顶点的下标
* @param weight 边的权值
*/
public void insertEdge(int v1, int v2, int weight) {
edges[v1][v2] = weight;
edges[v2][v1] = weight;
numberOfEdges++;
}
//对一个结点进行广度优先遍历的方法
public void bfs(boolean[] isVisited,int i){
int u;//表示队列的头结点对应下标
int w; //邻接结点w
//队列 记录结点访问的顺序
LinkedList queue = new LinkedList<>();
System.out.print(getValueByIndex(i)+"=>");
//标记为已访问
isVisited[i]=true;
//将结点加入队列
queue.addLast(i);
while (!queue.isEmpty()){
//取出队列的头结点的下标
u=(Integer)queue.removeFirst();
//得到第一个邻接点的下标w
w=getFirstNeighbor(u);
while (w!=-1){
//找到
//是否访问过
if (!isVisited[w]){
System.out.print(getValueByIndex(w)+"=>");
//标记已经访问过
isVisited[w]=true;
//入队列
queue.addLast(w);
}
//以u为前驱点,找w后面的下一个邻接点
w=getNextNeighbor(u,w);//体现出我们的广度优先
}
}
}
//遍历所有的结点,都进行广度优先搜索
public void bfs(){
isVisited=new boolean[8];
for (int i=0;i