赞
踩
本章源码:https://github.com/name365/Java-Data-structure
二分查找算法(非递归)介绍 前面我们讲过了二分查找算法,是使用递归的方式,下面我们讲解二分查找算法的非递归方式 二分查找法只适用于从有序的数列中进行查找(比如数字和字母等),将数列排序后再进行查找 二分查找法的运行时间为对数时间O(㏒₂n) ,即查找到需要的目标位置最多只需要㏒₂n步,假设从[0,99]的队列(100个数,即n=100)中寻到目标数30,则需要查找步数为㏒₂100 , 即最多需要查找7次( 2^6 < 100 < 2^7)
- 1
- 2
- 3
- 4
- 5
public class BinarySearchNoRecur { public static void main(String[] args) { int arr[] = {1,3, 8, 10, 11, 67, 100}; int index = bSart(arr, 100); System.out.println("index:" + index); } //二分查找算法 /** * * @Description * @author subei * @date 2020年6月20日下午3:41:33 * @param arr 待查找的数组, arr是升序排序 * @param target 需要查找的数 * @return 返回对应下标,-1表示没有找到 */ public static int bSart(int[] arr, int target) { int left =0; int right = arr.length - 1; while(left <= right){ //继续进行查找 int mid = (left+right)/2; if(arr[mid] == target){ return mid; }else if(arr[mid] > target){ right = mid - 1; //向左查找 }else{ left = mid + 1; //向右查找 } } return -1; //没找到 } }
分治算法介绍 分治法是一种很重要的算法。字面上的解释是“分而治之”,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并。这个技巧是很多高效算法的基础,如排序算法(快速排序,归并排序),傅立叶变换(快速傅立叶变换)…… 分治算法可以求解的一些经典问题: 二分搜索 大整数乘法 棋盘覆盖 合并排序 快速排序 线性时间选择 最接近点对问题 循环赛日程表 汉诺塔
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
分治(Divide-and-Conquer§)算法设计模式如下:
其中|P|表示问题P的规模;n0为一阈值,表示当问题P的规模不超过n0时,问题已容易直接解出,不必再继续分解。
ADHOC§是该分治法中的基本子算法,用于直接解小规模的问题P。
因此,当P的规模不超过n0时直接用算法ADHOC§求解。
算法MERGE(y1,y2,…,yk)是该分治法中的合并子算法,用于将P的子问题P1 ,P2 ,…,Pk的相应的解y1,y2,…,yk合并为P的解。
分治算法最佳实践——汉诺塔
汉诺塔的传说
汉诺塔:汉诺塔(又称河内塔)问题是源于印度一个古老传说的益智玩具。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。
假如每秒钟一次,共需多长时间呢?移完这些金片需要5845.54亿年以上,太阳系的预期寿命据说也就是数百亿年。真的过了5845.54亿年,地球上的一切生命,连同梵塔、庙宇等,都早已经灰飞烟灭。
汉诺塔游戏的演示和思路分析:
1)如果是有一个盘,A->C
2)如果我们有 n >= 2 情况,我们总是可以看做是两个盘:
1.最下边的盘
2.上面的盘
操作步骤:
1)先把最上面的盘 A->B
2)把最下边的盘 A->C
3)把B塔的所有盘 从 B->C
代码实现如下:
public class Hanoitower { public static void main(String[] args) { Htower(10,'A','P','Q'); } //汉诺塔算法 //用分治算法求解 public static void Htower(int num,char a,char b,char c){ //如果只有一个盘 if(num==1){ System.out.println("第一个盘从" + a + "->" + c); }else{ //如果我们有 n >= 2 情况,可以假设是两个盘 1.最下边的一个盘 2.上面的所有盘 //1.先把最上面的所有盘 A->B,移动过程会使用到 c Htower(num-1,a,c,b); //2.把最下边的盘 A->C System.out.println("第" + num + "个盘从 " + a + "->" + c); //3.把B塔的所有盘 从 B->C,移动过程使用到 a塔 Htower(num-1,b,a,c); } } }
物品 | 重量 | 价格 |
---|---|---|
吉他(G) | 1 | 1500 |
音响(S) | 4 | 3000 |
电脑(L) | 3 | 2000 |
动态规划算法介绍
1.动态规划(Dynamic Programming)算法的核心思想是:
将大问题划分为小问题进行解决,从而一步步获取最优解的处理算法
2.动态规划算法与分治算法类似,其基本思想也是
将待求解问题分解成若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解。
3.与分治法不同的是,适合于用动态规划求解的问题,经分解得到子问题往往不是互相独立的。
( 即下一个子阶段的求解是建立在上一个子阶段的解的基础上,进行进一步的求解 )
4.动态规划可以通过填表的方式来逐步推进,得到最优解.
思路分析和图解 背包问题主要是指一个给定容量的背包、若干具有一定价值和重量的物品,如何选择物品放入背包使物品的价值最大。 其中又分01背包和完全背包(完全背包指的是:每种物品都有无限件可用) 这里的问题属于01背包,即每个物品最多放一个。而无限背包可以转化为01背包。
- 1
- 2
- 3
- 4
算法的主要思想,利用动态规划来解决。 每次遍历到的第i个物品,根据w[i]和v[i]来确定是否需要将该物品放入背包中。 即对于给定的n个物品,设v[i]、w[i]分别为第i个物品的价值和重量,C为背包的容量。 再令v[i][j]表示在前i个物品中能够装入容量为j的背包中的最大价值。则我们有下面的结果: (1) v[i][0]=v[0][j]=0; //表示 填入表 第一行和第一列是0 (2) 当w[i]> j 时:v[i][j]=v[i-1][j] // 当准备加入新增的商品的容量大于 当前背包的容量时,就直接使用上一个单元格的装入策略 (3) 当j>=w[i]时: v[i][j]=max{v[i-1][j], v[i]+v[i-1][j-w[i]]} // 当 准备加入的新增的商品的容量小于等于当前背包的容量, // 装入的方式: v[i-1][j]: 就是上一个单元格的装入的最大值 v[i] : 表示当前商品的价值 v[i-1][j-w[i]] : 装入i-1商品,到剩余空间j-w[i]的最大值 当j>=w[i]时: v[i][j]=max{v[i-1][j], v[i]+v[i-1][j-w[i]]} :
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
上述三步公式进行验证:
验证1:v[1][1] = 1500
1. i = 1, j = 1
2. w[i] = w[1] = 1
w [1] = 1 j = 1 v[i][j]=max{v[i-1][j], v[i]+v[i-1][j-w[i]]} :
3. v[1][1] = max {v[0][1], v[1] + v[0][1-1]} = max{0, 1500 + 0} = 1500
验证2:v[3][4] = 2000+1500
1. i = 3;j = 4
2. w[i] = w[3] =3, j = 4
j = 4 >= w[i] = 3 => 4 >= 3
3. v[3][4] = max {v[2][4], v[3] + v[2][1]} = max{3000, 2000+1500} = 2000+1500
public class PackTest { public static void main(String[] args) { int[] arr = {1,4,3}; //物品重量 int[] var = {1500,3000,2000}; //物品的价值 int m = 4; //包的容量 int n = var.length; //物品的个数 //创建二维数组,表 //v[i][j] 表示在前i个物品中能够装入容量为j的背包中的最大价值 int[][] v = new int[n+1][m+1]; int[][] path = new int[n+1][m+1]; //记录放入商品的情况 //初始化第一行和第一列,本程序可不做此步 for(int i = 0;i < v.length;i++){ v[i][0]=0; //将第一列设为0 } for(int i = 0;i < v[0].length;i++){ v[0][i]=0; //将第一列设为0 } //根据相关公式进行动态处理 for(int i = 1;i < v.length;i++){ //不处理第一行 for(int j = 1;j < v[0].length;j++){ //不处理第一列 if(arr[i-1]> j) { // 因为程序i是从1开始的,因此原来公式中的 arr[i],修改成 arr[i-1] v[i][j]=v[i-1][j]; } else { //说明: //因为i是从1开始的,因此公式需要调整成 //v[i][j]=Math.max(v[i-1][j], var[i-1]+v[i-1][j-w[i-1]]); //v[i][j] = Math.max(v[i - 1][j], var[i - 1] + v[i - 1][j - w[i - 1]]); if(v[i - 1][j] < var[i - 1] + v[i - 1][j - arr[i - 1]]) { v[i][j] = var[i - 1] + v[i - 1][j - arr[i - 1]]; //把当前的情况记录到path path[i][j] = 1; } else { v[i][j] = v[i - 1][j]; } } } } //输出 for(int i = 0;i < v.length;i++){ for(int j = 0;j < v[i].length;j++){ System.out.print(v[i][j] + " "); } System.out.println(); } //输出最后放入的哪些商品 // for(int i=0;i< path.length;i++){ // for(int j=0;j< path[i].length;j++){ // if(path[i][j] == 1){ // System.out.printf("第%d个商品放入到背包\n", i); // } // } // } int i = path.length - 1; //行的最大下标 int j = path[0].length - 1; //列的最大下标 while(i > 0 && j > 0 ) { //从path的最后开始找 if(path[i][j] == 1) { System.out.printf("第%d个商品放入到背包\n", i); j -= arr[i-1]; //arr[i-1] } i--; } } }
应用场景——字符串匹配问题
暴力匹配算法
public class ViolenceMatch { public static void main(String[] args) { String str1 = "数据 数据结构 数据结构与算法 Java数据结构"; String str2 ="数据结构"; int index = violenceMatch(str1, str2); System.out.println("index=" + index); } // 暴力匹配算法实现 public static int violenceMatch(String str1, String str2) { char[] s1 = str1.toCharArray(); char[] s2 = str2.toCharArray(); int s1Len = s1.length; int s2Len = s2.length; int i = 0; // i索引指向s1 int j = 0; // j索引指向s2 while (i < s1Len && j < s2Len) { // 保证匹配不会越界 if (s1[i] == s2[j]) { // 匹配成功 i++; j++; } else { // 没有匹配成功 // 如果配失败(即str1[i]! = str2[j]),令i = i - (j - 1),j = 0 i = i - (j - 1); j = 0; } } // 判断是否匹配成功 if (j == s2Len) { return i - j; } else { return -1; } } }
(其实是很不明智的,因为此时BCD已经比较过了,没有必要再做重复的工作,一个基本事实是,当空格与D不匹配时,你其实知道前面六个字符是”ABCDAB”。KMP 算法的想法是,设法利用这个已知信息,不要把”搜索位置”移回已经比较过的位置,继续把它向后移,这样就提高了效率。)
7.怎么做到把刚刚重复的步骤省略掉?可以对Str2计算出一张《部分匹配表》,这张表的产生在后面介绍
移动位数 = 已匹配的字符数 - 对应的部分匹配值
因为 6 - 2 等于4,所以将搜索词向后移动 4 位。
到此KMP算法思想分析完毕!
具体代码实现:
import java.util.Arrays; public class KMPTest { public static void main(String[] args) { String str1 = "BBC ABCDAB ABCDABCDABDE"; String str2 = "ABCDABD"; // String str2 = "BBC"; int[] next = KmpT("ABCDABD"); System.out.println("neat:" + Arrays.toString(next)); int index = kmpSearch(str1,str2,next); System.out.println("index:" + index); } //KMP搜索算法 /** * * @Description * @author subei * @date 2020年6月22日下午6:47:04 * @param str1 源字符串 * @param str2 字串 * @param next 部分匹配表,子串所对应的部分匹配表 * @return 如果是-1就是没有匹配到,否则返回第一个匹配的位置 */ public static int kmpSearch(String str1,String str2,int[] next){ for(int i=0,j=0;i < str1.length();i++){ //调整j的大小 while(j > 0 && str1.charAt(i) != str2.charAt(j)){ j = next[j-1]; } if(str1.charAt(i) == str2.charAt(j)){ j++; } if(j == str2.length()){ //找到了!!! return i - j + 1; } } return -1; } //获取一个字符串(子串) 的部分匹配值表 public static int[] KmpT(String dest){ //创建一个next数组保存部分匹配值 int[] next = new int[dest.length()]; next[0] = 0; //如果字符串是长度为1 部分匹配值就是0 for(int i = 1, j = 0; i < dest.length(); i++) { //当dest.charAt(i) != dest.charAt(j) ,此时需要从next[j-1]获取新的j //直到被发现有 dest.charAt(i) == dest.charAt(j)成立才退出 //这是kmp算法的核心点!!!! while(j > 0 && dest.charAt(i) != dest.charAt(j)) { j = next[j-1]; } //当dest.charAt(i)==dest.charAt(j) 满足时,部分匹配值就+1 if(dest.charAt(i) == dest.charAt(j)) { j++; } next[i] = j; } return next; } }
应用场景
假设存在下面需要付费的广播台,以及广播台信号可以覆盖的地区。 如何选择最少的广播台,让所有的地区都可以接收到信号。
广播台 | 覆盖地区 |
---|---|
K1 | “北京”, “上海”, “天津” |
K2 | “广州”, “北京”, “深圳” |
K3 | “成都”, “上海”, “杭州” |
K4 | “上海”, “天津” |
K5 | “杭州”, “大连” |
具体介绍:
思路分析:
广播台数量n | 子集总数2ⁿ | 需要的时间 |
---|---|---|
5 | 32 | 3.2秒 |
10 | 1024 | 102.4秒 |
32 | 4294967296 | 13.6年 |
100 | 1.26*100³º | 4x10²³年 |
import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; public class GreadeTest { public static void main(String[] args) { // 创建电台,存入Map HashMap<String, HashSet<String>> broadcasts = new HashMap<String, HashSet<String>>(); // 将各电台放入broadcasts HashSet<String> hashSet1 = new HashSet<String>(); hashSet1.add("北京"); hashSet1.add("上海"); hashSet1.add("天津"); HashSet<String> hashSet2 = new HashSet<String>(); hashSet2.add("广州"); hashSet2.add("北京"); hashSet2.add("深圳"); HashSet<String> hashSet3 = new HashSet<String>(); hashSet3.add("成都"); hashSet3.add("上海"); hashSet3.add("杭州"); HashSet<String> hashSet4 = new HashSet<String>(); hashSet4.add("上海"); hashSet4.add("天津"); HashSet<String> hashSet5 = new HashSet<String>(); hashSet5.add("杭州"); hashSet5.add("大连"); // 加入到map broadcasts.put("K1", hashSet1); broadcasts.put("K2", hashSet2); broadcasts.put("K3", hashSet3); broadcasts.put("K4", hashSet4); broadcasts.put("K5", hashSet5); // allAreas存放所有的地区 HashSet<String> allAreas = new HashSet<String>(); allAreas.add("北京"); allAreas.add("上海"); allAreas.add("天津"); allAreas.add("广州"); allAreas.add("深圳"); allAreas.add("成都"); allAreas.add("杭州"); allAreas.add("大连"); // 创建ArrayList,存放选择的电台 ArrayList<String> selects = new ArrayList<String>(); // 定义一个临时的集合,在遍历的过程中,存放遍历过程中的电台覆盖的地区和当前还没有覆盖的地区的交集 HashSet<String> tempSet = new HashSet<String>(); // 定义给maxKey,保存在一次遍历过程中,能够覆盖最大未覆盖的地区对应的电台的key // 如果maxKey不为null,则会加入到 selects String maxKey = null; //保存临时变量 while (allAreas.size() != 0) { // 如果allAreas不为0, 则表示还没有覆盖到所有的地区 // 每进行一次while,需要 maxKey = null; // 遍历 broadcasts, 取出对应key for (String key : broadcasts.keySet()) { // 每进行一次for tempSet.clear(); // 当前这个key能够覆盖的地区 HashSet<String> areas = broadcasts.get(key); tempSet.addAll(areas); // 求出tempSet 和 allAreas 集合的交集, 交集会赋给 tempSet tempSet.retainAll(allAreas); // 如果当前这个集合包含的未覆盖地区的数量,比maxKey指向的集合地区还多 // 就需要重置maxKey // tempSet.size() > broadcasts.get(maxKey).size()) // 体现出贪心算法的特点,每次都选择最优的 if (tempSet.size() > 0 && (maxKey == null || tempSet.size() > broadcasts.get(maxKey).size())) { maxKey = key; } } // maxKey != null, 就应该将maxKey 加入selects if (maxKey != null) { selects.add(maxKey); // 将maxKey指向的广播电台覆盖的地区,从 allAreas 去掉 allAreas.removeAll(broadcasts.get(maxKey)); } } System.out.println("得到的选择结果是" + selects);// [K1,K2,K3,K5] } }
贪心算法注意事项和细节:
先看一个应用场景和问题:
有胜利乡有7个村庄(A, B, C, D, E, F, G) ,现在需要修路把7个村庄连通
各个村庄的距离用边线表示(权) ,比如 A – B 距离 5公里
问:如何修路保证各个村庄都能连通,并且总的修建公路总里程最短?
思路: 将10条边,连接即可,但是总的里程数不是最小.
正确的思路,就是尽可能的选择少的路线,并且每条路线最小,保证总里程数最少.
最小生成树:
修路问题本质就是就是最小生成树问题, 先介绍一下最小生成树(Minimum Cost Spanning Tree),简称MST。
给定一个带权的无向连通图,如何选取一棵生成树,使树上所有边上权的总和为最小,这叫最小生成树
N个顶点,一定有N-1条边
包含全部顶点
N-1条边都在图中
举例说明(如图:)
求最小生成树的算法主要是普里姆算法和克鲁斯卡尔算法
普利姆算法介绍:
普利姆(Prim)算法求最小生成树,也就是在包含n个顶点的连通图中,找出只有(n-1)条边包含所有n个顶点的连通子图,也就是所谓的极小连通子图
普利姆的算法如下:
设G=(V,E)是连通网,T=(U,D)是最小生成树,V,U是顶点集合,E,D是边的集合
若从顶点u开始构造最小生成树,则从集合V中取出顶点u放入集合U中,标记顶点v的visited[u]=1
若集合U中顶点ui与集合V-U中的顶点vj之间存在边,则寻找这些边中权值最小的边,但不能构成回路,将顶点vj加入集合U中,将边(ui,vj)加入集合D中,标记visited[vj]=1
重复步骤②,直到U与V相等,即所有顶点都被标记为访问过,此时D中有n-1条边
提示: 单独看步骤很难理解,我们通过代码来讲解,比较好理解.
1.从<A>顶点开始处理 ======> <A,G> 2
A-C [7] A-G[2] A-B[5] =>
2. <A,G> 开始 , 将A 和 G 顶点和他们相邻的还没有访问的顶点进行处理 =》<A,G,B>
A-C[7] A-B[5] G-B[3] G-E[4] G-F[6]
3. <A,G,B> 开始,将A,G,B 顶点 和他们相邻的还没有访问的顶点进行处理=><A,G,B,E>
A-C[7] G-E[4] G-F[6] B-D[9]
.....
4.{A,G,B,E}->F//第4次大循环 , 对应 边<E,F> 权值:5
5.{A,G,B,E,F}->D//第5次大循环 , 对应 边<F,D> 权值:4
6. {A,G,B,E,F,D}->C//第6次大循环 , 对应 边<A,C> 权值:7 ===> <A,G,B,E,F,D,C>
代码实现如下:
import java.util.Arrays; public class PrimAlgorithm { public static void main(String[] args) { char[] data = new char[] { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; int verxs = data.length; // 邻接矩阵的关系使用二维数组表示,10000这个大数,表示两个点不联通 int[][] weight = new int[][] { { 10000, 5, 7, 10000, 10000, 10000, 2 }, { 5, 10000, 10000, 9, 10000, 10000, 3 }, { 7, 10000, 10000, 10000, 8, 10000, 10000 }, { 10000, 9, 10000, 10000, 10000, 4, 10000 }, { 10000, 10000, 8, 10000, 10000, 5, 4 }, { 10000, 10000, 10000, 4, 5, 10000, 6 }, { 2, 3, 10000, 10000, 4, 6, 10000 }, }; // 创建MGraph对象 MGraph graph = new MGraph(verxs); // 创建一个MinTree对象 MinTree minTree = new MinTree(); minTree.createGraph(graph, verxs, data, weight); // 输出 minTree.showGraph(graph); //测试普利姆算法 minTree.prim(graph, 0); } } // 创建最小生成树->村庄的图 class MinTree { // 创建图的邻接矩阵 /** * * @Description * @author subei * @date 2020年6月23日下午3:54:19 * @param graph 图对象 * @param verxs 图对应的顶点个数 * @param data 图的各个顶点的值 * @param weight 图的邻接矩阵 */ public void createGraph(MGraph graph, int verxs, char data[], int[][] weight) { int i, j; for (i = 0; i < verxs; i++) {// 顶点 graph.data[i] = data[i]; for (j = 0; j < verxs; j++) { graph.weight[i][j] = weight[i][j]; } } } // 显示图的邻接矩阵 public void showGraph(MGraph graph) { for (int[] link : graph.weight) { System.out.println(Arrays.toString(link)); } } // 编写prim算法,得到最小生成树 /** * * @Description * @author subei * @date 2020年6月23日下午5:52:27 * @param graph 图 * @param v 表示从图的第几个顶点开始生成 */ public void prim(MGraph graph, int v) { // visited[] 标记结点(顶点)是否被访问过 int visited[] = new int[graph.verxs]; // visited[] 默认元素的值都是0, 表示没有访问过 // for(int i =0; i <graph.verxs; i++) { // visited[i] = 0; // } // 把当前这个结点标记为已访问 visited[v] = 1; // h1和h2记录两个顶点的下标 int h1 = -1; int h2 = -1; int minWeight = 10000; // 将 minWeight 初始成一个大数,后面在遍历过程中,会被替换 for (int k = 1; k < graph.verxs; k++) {// 因为有 graph.verxs顶点,普利姆算法结束后,有 // graph.verxs-1边 // 此处是确定每一次生成的子图,和哪个结点的距离最近 for (int i = 0; i < graph.verxs; i++) { // i结点表示被访问过的结点 for (int j = 0; j < graph.verxs; j++) { // j结点表示还没有访问过的结点 if (visited[i] == 1 && visited[j] == 0 && graph.weight[i][j] < minWeight) { // 替换minWeight(寻找已经访问过的结点和未访问过的结点间的权值最小的边) minWeight = graph.weight[i][j]; h1 = i; h2 = j; } } } // 找到一条边是最小 System.out.println("边<" + graph.data[h1] + "," + graph.data[h2] + "> 权值:" + minWeight); // 将当前这个结点标记为已经访问 visited[h2] = 1; // minWeight 重新设置为最大值 10000 minWeight = 10000; } } } class MGraph { int verxs; // 图的节点个数 char[] data; // 存放节点数据 int[][] weight; // 存放边 public MGraph(int verxs) { this.verxs = verxs; data = new char[verxs]; weight = new int[verxs][verxs]; } }
看一个应用场景和问题:
在含有n个顶点的连通图中选择n-1条边,构成一棵极小连通子图,并使该连通子图中n-1条边上权值之和达到最小,则称其为连通网的最小生成树。
第1步:将边<E,F>加入R中。
边<E,F>的权值最小,因此将它加入到最小生成树结果R中。
第2步:将边<C,D>加入R中。
上一步操作之后,边<C,D>的权值最小,因此将它加入到最小生成树结果R中。
第3步:将边<D,E>加入R中。
上一步操作之后,边<D,E>的权值最小,因此将它加入到最小生成树结果R中。
第4步:将边<B,F>加入R中。
上一步操作之后,边<C,E>的权值最小,但<C,E>会和已有的边构成回路;因此,跳过边<C,E>。同理,跳过边<C,F>。将边<B,F>加入到最小生成树结果R中。
第5步:将边<E,G>加入R中。
上一步操作之后,边<E,G>的权值最小,因此将它加入到最小生成树结果R中。
第6步:将边<A,B>加入R中。
上一步操作之后,边<F,G>的权值最小,但<F,G>会和已有的边构成回路;因此,跳过边<F,G>。同理,跳过边<B,C>。将边<A,B>加入到最小生成树结果R中。
此时,最小生成树构造完成!它包括的边依次是:<E,F> <C,D> <D,E> <B,F> <E,G> <A,B>。
根据前面介绍的克鲁斯卡尔算法的基本思想和做法,我们能够了解到,克鲁斯卡尔算法重点需要解决的以下两个问题:
问题一很好解决,采用排序算法进行排序即可。
问题二,处理方式是:记录顶点在"最小生成树"中的终点,顶点的终点是"在最小生成树中与它连通的最大顶点"。然后每次需要将一条边添加到最小生存树时,判断该边的两个顶点的终点是否重合,重合的话则会构成回路。
在将<E,F> <C,D> <D,E>加入到最小生成树R中之后,这几条边的顶点就都有了终点:
(01) C的终点是F。
(02) D的终点是F。
(03) E的终点是F。
(04) F的终点是F。
import java.util.Arrays; public class KruskalCase { private int edgeNum; // 边的个数 private char[] vertexs; // 顶点数组 private int[][] matrix; // 邻接矩阵 // 使用 INF,表示两个顶点不能连通 private static final int INF = Integer.MAX_VALUE; public static void main(String[] args) { char[] vertexs = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; //克鲁斯卡尔算法的邻接矩阵 int matrix[][] = { /*A*//*B*//*C*//*D*//*E*//*F*//*G*/ /*A*/ { 0, 12, INF, INF, INF, 16, 14}, /*B*/ { 12, 0, 10, INF, INF, 7, INF}, /*C*/ { INF, 10, 0, 3, 5, 6, INF}, /*D*/ { INF, INF, 3, 0, 4, INF, INF}, /*E*/ { INF, INF, 5, 4, 0, 2, 8}, /*F*/ { 16, 7, 6, INF, 2, 0, 9}, /*G*/ { 14, INF, INF, INF, 8, 9, 0}}; //创建KruskalCase 对象实例 KruskalCase kruskalCase = new KruskalCase(vertexs, matrix); //输出构建的 kruskalCase.print(); kruskalCase.kruskal(); } // 构造器 public KruskalCase(char[] vertexs, int[][] matrix) { // 初始化顶点数与边的个数 int vlen = vertexs.length; // 初始化顶点,复制拷贝的方式 this.vertexs = new char[vlen]; for (int i = 0; i < vertexs.length; i++) { this.vertexs[i] = vertexs[i]; } // 初始化边,使用的是复制拷贝的方式 this.matrix = new int[vlen][vlen]; for (int i = 0; i < vlen; i++) { for (int j = 0; j < vlen; j++) { this.matrix[i][j] = matrix[i][j]; } } // 统计边的条数 for (int i = 0; i < vlen; i++) { for (int j = i + 1; j < vlen; j++) { if (this.matrix[i][j] != INF) { edgeNum++; } } } } //打印邻接矩阵 public void print(){ System.out.println("邻接矩阵:"); for(int i = 0;i < vertexs.length;i++){ for(int j = 0;j< vertexs.length;j++){ System.out.printf("%10d\t",matrix[i][j]); } System.out.println(); } } /** * * @Description 对边进行排序处理,冒泡排序 * @author subei * @param edges 边的集合 */ private void sortEdges(EData[] edges) { for (int i = 0; i < edges.length - 1; i++) { for (int j = 0; j < edges.length - 1 - i; j++) { if (edges[j].weight > edges[j + 1].weight) {// 交换 EData tmp = edges[j]; edges[j] = edges[j + 1]; edges[j + 1] = tmp; } } } } /** * * @Description * @author subei * @param ch 顶点的值,比如'A','B' * @return 返回ch顶点对应的下标,如果找不到,返回-1 */ private int getPosition(char ch) { for (int i = 0; i < vertexs.length; i++) { if (vertexs[i] == ch) {// 找到 return i; } } // 找不到,返回-1 return -1; } /** * * @Description 获取图中边,放到EData[] 数组中,因为需要遍历该数组 * 是通过matrix 邻接矩阵来获取 * EData[] 形式 [['A','B', 12], ['B','F',7], .....] * @author subei * @return */ private EData[] getEdges() { int index = 0; EData[] edges = new EData[edgeNum]; for (int i = 0; i < vertexs.length; i++) { for (int j = i + 1; j < vertexs.length; j++) { if (matrix[i][j] != INF) { edges[index++] = new EData(vertexs[i], vertexs[j], matrix[i][j]); } } } return edges; } /** * i = 4 [0,0,0,0,5,0,0,0,0,0,0,0] * @Description 获取下标为i的顶点的终点(), 用于判断两个顶点的终点是否相同 * @author subei * @param ends 数组就是记录了各个顶点对应的终点是哪个,ends 数组是在遍历过程中,逐步形成 * @param i 表示传入的顶点对应的下标 * @return 返回的就是下标为i的这个顶点对应的终点的下标 */ private int getEnd(int[] ends, int i) { while (ends[i] != 0) { i = ends[i]; } return i; } public void kruskal() { int index = 0; //表示最后结果数组的索引 int[] ends = new int[edgeNum]; //用于保存"已有最小生成树" 中的每个顶点在最小生成树中的终点 //创建结果数组, 保存最后的最小生成树 EData[] rets = new EData[edgeNum]; //获取图中 所有的边的集合 , 一共有12边 EData[] edges = getEdges(); System.out.println("图的边的集合:" + Arrays.toString(edges) + " 共"+ edges.length); //12 //按照边的权值大小进行排序(从小到大) sortEdges(edges); //遍历edges 数组,将边添加到最小生成树中时,判断是准备加入的边否形成了回路,如果没有,就加入 rets, 否则不能加入 for(int i=0; i < edgeNum; i++) { //获取到第i条边的第一个顶点(起点) int p1 = getPosition(edges[i].start); //p1=4 //获取到第i条边的第2个顶点 int p2 = getPosition(edges[i].end); //p2 = 5 //获取p1这个顶点在已有最小生成树中的终点 int m = getEnd(ends, p1); //m = 4 //获取p2这个顶点在已有最小生成树中的终点 int n = getEnd(ends, p2); // n = 5 //是否构成回路 if(m != n) { //没有构成回路 ends[m] = n; // 设置m 在"已有最小生成树"中的终点 <E,F> [0,0,0,0,5,0,0,0,0,0,0,0] rets[index++] = edges[i]; //有一条边加入到rets数组 } } //<E,F> <C,D> <D,E> <B,F> <E,G> <A,B>。 //统计并打印 "最小生成树", 输出 rets System.out.println("最小生成树为:"); for(int i = 0; i < index; i++) { System.out.println(rets[i]); } } } //创建一个类EData,它的对象实例就表示一条边 class EData { char start; // 边的一个点 char end; // 边的另外一个点 int weight; // 边的权值 // 构造器 public EData(char start, char end, int weight) { super(); this.start = start; this.end = end; this.weight = weight; } // 输出边信息 @Override public String toString() { return "EData [<" + start + ", " + end + ">= " + weight + "]"; } }
看一个应用场景和问题:
迪杰斯特拉(Dijkstra)算法是典型最短路径算法,用于计算一个结点到其他结点的最短路径。 它的主要特点是以起始点为中心向外层层扩展(广度优先搜索思想),直到扩展到终点为止。
迪杰斯特拉(Dijkstra)算法过程
设置出发顶点为v,顶点集合V{v1,v2,vi...},v到V中各顶点的距离构成距离集合Dis,Dis{d1,d2,di...},Dis集合记录着v到图中各顶点的距离(到自身可以看作0,v到vi距离对应为di)
从Dis中选择值最小的di并移出Dis集合,同时移出V集合中对应的顶点vi,此时的v到vi即为最短路径
更新Dis集合,更新规则为:比较v到V集合中顶点的距离值,与v通过vi到V集合中顶点的距离值,保留值较小的一个(同时也应该更新顶点的前驱节点为vi,表明是通过vi到达的)
重复执行两步骤,直到最短路径顶点为目标顶点即可结束
代码实现如下:
import java.util.Arrays; public class DijkstraAlgorithm { public static void main(String[] args) { char[] vertex = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; // 邻接矩阵 int[][] matrix = new int[vertex.length][vertex.length]; final int N = 65535;// 表示不可以连接 matrix[0] = new int[] { N, 5, 7, N, N, N, 2 }; matrix[1] = new int[] { 5, N, N, 9, N, N, 3 }; matrix[2] = new int[] { 7, N, N, N, 8, N, N }; matrix[3] = new int[] { N, 9, N, N, N, 4, N }; matrix[4] = new int[] { N, N, 8, N, N, 5, 4 }; matrix[5] = new int[] { N, N, N, 4, 5, N, 6 }; matrix[6] = new int[] { 2, 3, N, N, 4, 6, N }; // 创建 Graph对象 Graph graph = new Graph(vertex, matrix); // 测试图的邻接矩阵 graph.showGraph(); //测试迪杰斯特拉算法 graph.dsj(2); graph.showDijkstra(); } } class Graph { private char[] vertex; // 顶点数组 private int[][] matrix; // 邻接矩阵 private VisitedVertex vv; // 已经访问的顶点的集合 // 构造器 public Graph(char[] vertex, int[][] matrix) { super(); this.vertex = vertex; this.matrix = matrix; } // 显示图 public void showGraph() { for (int[] link : matrix) { System.out.println(Arrays.toString(link)); } } // 显示结果 public void showDijkstra() { vv.show(); } /** * @Description 迪杰斯特拉算法实现 * @author subei * @param index 表示出发顶点对应的下标 */ public void dsj(int index) { vv = new VisitedVertex(vertex.length, index); update(index);// 更新index顶点到周围顶点的距离和前驱顶点 for (int j = 1; j < vertex.length; j++) { index = vv.updateArr();// 选择并返回新的访问顶点 update(index); // 更新index顶点到周围顶点的距离和前驱顶点 } } // 更新index下标顶点到周围顶点的距离和周围顶点的前驱顶点 private void update(int index) { int len = 0; // 根据遍历我们的邻接矩阵的 matrix[index]行 for (int j = 0; j < matrix[index].length; j++) { // len 含义是: 出发顶点到index顶点的距离 + 从index顶点到j顶点的距离的和 len = vv.getDis(index) + matrix[index][j]; // 如果j顶点没有被访问过,并且 len 小于出发顶点到j顶点的距离,就需要更新 if (!vv.in(j) && len < vv.getDis(j)) { vv.updatePre(j, index); // 更新j顶点的前驱为index顶点 vv.updateDis(j, len); // 更新出发顶点到j顶点的距离 } } } } // 已访问顶点集合 class VisitedVertex { // 记录各个顶点是否访问过 1表示访问过,0未访问,会动态更新 public int[] already_arr; // 每个下标对应的值为前一个顶点下标, 会动态更新 public int[] pre_visited; // 记录出发顶点到其他所有顶点的距离,比如G为出发顶点,就会记录G到其它顶点的距离,会动态更新,求的最短距离就会存放到dis public int[] dis; /** * @param length 表示顶点的个数 * @param index 出发顶点对应的下标, 比如G顶点,下标就是6 */ public VisitedVertex(int length, int index) { this.already_arr = new int[length]; this.pre_visited = new int[length]; this.dis = new int[length]; // 初始化 dis数组 Arrays.fill(dis, 65535); this.already_arr[index] = 1; // 设置出发顶点被访问过 this.dis[index] = 0;// 设置出发顶点的访问距离为0 } /** * * @Description 判断index顶点是否被访问过 * @author subei * @param index * @return 如果访问过,就返回true, 否则访问false */ public boolean in(int index) { return already_arr[index] == 1; } //更新出发顶点到index顶点的距离 public void updateDis(int index, int len) { dis[index] = len; } //更新pre这个顶点的前驱顶点为index顶点 public void updatePre(int pre, int index) { pre_visited[pre] = index; } //返回出发顶点到index顶点的距离 public int getDis(int index) { return dis[index]; } //继续选择并返回新的访问顶点,比如这里的G完后,就是 A点作为新的访问顶点(注意不是出发顶点) public int updateArr() { int min = 65535, index = 0; for (int i = 0; i < already_arr.length; i++) { if (already_arr[i] == 0 && dis[i] < min) { min = dis[i]; index = i; } } // 更新 index 顶点被访问过 already_arr[index] = 1; return index; } // 显示最后的结果:即将三个数组的情况输出 public void show() { System.out.println("++++++++++++++++++++++"); // 输出already_arr for (int i : already_arr) { System.out.print(i + " "); } System.out.println(); // 输出pre_visited for (int i : pre_visited) { System.out.print(i + " "); } System.out.println(); // 输出dis for (int i : dis) { System.out.print(i + " "); } System.out.println(); // 为了美观最后的最短距离,处理 char[] vertex = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; int count = 0; for (int i : dis) { if (i != 65535) { System.out.print(vertex[count] + "(" + i + ") "); } else { System.out.println("N "); } count++; } System.out.println(); } }
相关简介:
和Dijkstra算法一样,弗洛伊德(Floyd)算法也是一种用于寻找给定的加权图中顶点间最短路径的算法。该算法名称以创始人之一、1978年图灵奖获得者、斯坦福大学计算机科学系教授罗伯特·弗洛伊德命名
弗洛伊德算法(Floyd)计算图中各个顶点之间的最短路径
迪杰斯特拉算法用于计算图中某一个顶点到其他顶点的最短路径。
弗洛伊德算法 VS 迪杰斯特拉算法:迪杰斯特拉算法通过选定的被访问顶点,求出从出发访问顶点到其他顶点的最短路径;弗洛伊德算法中每一个顶点都是出发访问点,所以需要将每一个顶点看做被访问顶点,求出从每一个顶点到其他顶点的最短路径。
弗洛伊德(Floyd)算法图解分析:
距离表的数据来源: A A A F G G A (A到A的最短路径是0) (A到B的最短路径是5) (A到C的最短路径是7) (A到D的最短路径是12) (A到E的最短路径是6) (A到F的最短路径是8) (A到G的最短路径是2) B B A B G G B (B到A的最短路径是5) (B到B的最短路径是0) (B到C的最短路径是12) (B到D的最短路径是9) (B到E的最短路径是7) (B到F的最短路径是9) (B到G的最短路径是3) C A C F C E A (C到A的最短路径是7) (C到B的最短路径是12) (C到C的最短路径是0) (C到D的最短路径是17) (C到E的最短路径是8) (C到F的最短路径是13) (C到G的最短路径是9) G D E D F D F (D到A的最短路径是12) (D到B的最短路径是9) (D到C的最短路径是17) (D到D的最短路径是0) (D到E的最短路径是9) (D到F的最短路径是4) (D到G的最短路径是10) G G E F E E E (E到A的最短路径是6) (E到B的最短路径是7) (E到C的最短路径是8) (E到D的最短路径是9) (E到E的最短路径是0) (E到F的最短路径是5) (E到G的最短路径是4) G G E F F F F (F到A的最短路径是8) (F到B的最短路径是9) (F到C的最短路径是13) (F到D的最短路径是4) (F到E的最短路径是5) (F到F的最短路径是0) (F到G的最短路径是6) G G A F G G G (G到A的最短路径是2) (G到B的最短路径是3) (G到C的最短路径是9) (G到D的最短路径是10) (G到E的最短路径是4) (G到F的最短路径是6) (G到G的最短路径是0)
具体代码实现:
import java.util.Arrays; public class FloydAlgorithm { public static void main(String[] args) { char[] vertex = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; // 创建邻接矩阵 int[][] matrix = new int[vertex.length][vertex.length]; final int N = 65535; matrix[0] = new int[] { 0, 5, 7, N, N, N, 2 }; matrix[1] = new int[] { 5, 0, N, 9, N, N, 3 }; matrix[2] = new int[] { 7, N, 0, N, 8, N, N }; matrix[3] = new int[] { N, 9, N, 0, N, 4, N }; matrix[4] = new int[] { N, N, 8, N, 0, 5, 4 }; matrix[5] = new int[] { N, N, N, 4, 5, 0, 6 }; matrix[6] = new int[] { 2, 3, N, N, 4, 6, 0 }; // 创建 Graph 对象 Graph graph = new Graph(vertex, matrix, vertex.length); // 调用弗洛伊德算法 graph.floyd(); graph.show(); } } // 创建图 class Graph { @SuppressWarnings("unused") private char[] vertex; // 存放顶点的数组 private int[][] dis; // 保存,从各个顶点出发到其它顶点的距离,最后的结果,也是保留在该数组 private int[][] pre;// 保存到达目标顶点的前驱顶点 /** * @param vertex 顶点数组 * @param matrix 邻接矩阵 * @param length 大小 */ public Graph(char[] vertex, int[][] matrix, int length) { super(); this.vertex = vertex; this.dis = matrix; this.pre = new int[length][length]; // 对pre数组初始化,注意存放的是前驱顶点的下标 for (int i = 0; i < length; i++) { Arrays.fill(pre[i], i); } } // 显示pre数组和dis数组 public void show() { char[] vertex = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' }; for (int k = 0; k < dis.length; k++) { // 先将pre数组输出的一行 for (int i = 0; i < dis.length; i++) { System.out.print(vertex[pre[k][i]] + " "); } System.out.println(); // 输出dis数组的一行数据 for (int i = 0; i < dis.length; i++) { System.out.print("(" + vertex[k] + "到" + vertex[i] + "的最短路径是" + dis[k][i] + ") "); } System.out.println(); System.out.println(); } } // 弗洛伊德算法 public void floyd() { int len = 0; // 变量保存距离 // 对中间顶点遍历, k 就是中间顶点的下标 [A, B, C, D, E, F, G] for (int k = 0; k < dis.length; k++) { // // 从i顶点开始出发 [A, B, C, D, E, F, G] for (int i = 0; i < dis.length; i++) { // 到达j顶点 [A, B, C, D, E, F, G] for (int j = 0; j < dis.length; j++) { len = dis[i][k] + dis[k][j];// => 求出从i 顶点出发,经过 k中间顶点,到达 j // 顶点距离 if (len < dis[i][j]) {// 如果len小于 dis[i][j] dis[i][j] = len;// 更新距离 pre[i][j] = pre[k][j];// 更新前驱顶点 } } } } } }
算法简介:
骑士周游问题的解决步骤和思路: 1. 创建棋盘 chessBoard , 是一个二维数组 2. 将当前位置设置为已经访问,然后根据当前位置,计算马儿还能走哪些位置,并放入到一个集合中(ArrayList), 最多有8个位置, 每走一步,就使用step+1 3. 遍历ArrayList中存放的所有位置,看看哪个可以走通 , 如果走通,就继续,走不通,就回溯. 4. 判断马儿是否完成了任务,使用 step 和应该走的步数比较 , 如果没有达到数量,则表示没有完成任务,将整个棋盘置0 注意:马儿不同的走法(策略),会得到不同的结果,效率也会有影响(优化) //创建一个Point Point p1 = new Point(); if((p1.x = curPoint.x - 2) >= 0 && (p1.y = curPoint.y -1) >= 0) { ps.add(new Point(p1)); } 使用贪心算法对原来的算法优化 1。 我们获取当前位置,可以走的下一个位置的集合 //获取当前位置可以走的下一个位置的集合 ArrayList<Point> ps = next(new Point(column, row)); 2. 我们需要对 ps 中所有的Point 的下一步的所有集合的数目,进行非递减排序,就ok , 9, 7, 6, 5, 3, 2 , 1 //递减排序 1, 2, 3, 4,5,6, 10, //递增排序 1, 2, 2, 2, 3,3, 4, 5, 6 // 非递减 9, 7, 6,6, 6, 5,5, 3, 2 , 1 //非递增
代码实现如下:
import java.awt.Point; import java.util.ArrayList; import java.util.Comparator; public class HorseChessboard { private static int X; // 棋盘的列数 private static int Y; // 棋盘的行数 //创建一个数组,标记棋盘的各个位置是否被访问过 private static boolean visited[]; //使用一个属性,标记是否棋盘的所有位置都被访问 private static boolean finished; // 如果为true,表示成功 public static void main(String[] args) { System.out.println("骑士周游算法:"); X = 8; Y = 8; int row = 1; //马儿初始位置的行,从1开始编号 int column = 1; //马儿初始位置的列,从1开始编号 //创建棋盘 int[][] chessboard = new int[X][Y]; visited = new boolean[X * Y];//初始值为false //测试耗时 long start = System.currentTimeMillis(); traversalChessboard(chessboard, row - 1, column - 1, 1); long end = System.currentTimeMillis(); System.out.println("共耗时: " + (end - start) + " 毫秒"); //输出棋盘 for(int[] rows : chessboard) { for(int step: rows) { System.out.print(step + "\t"); } System.out.println(); } } /** * @Description 骑士周游问题的算法 * @author subei * @date 2020年6月27日上午10:58:06 * @param chessboard 棋盘 * @param row 马儿当前的位置的行 从0开始 * @param column 马儿当前的位置的列 从0开始 * @param step 是第几步,初始位置就是第1步 */ public static void traversalChessboard(int[][] chessboard, int row, int column, int step) { chessboard[row][column] = step; //row = 4 X = 8 column = 4 = 4 * 8 + 4 = 36 visited[row * X + column] = true; //标记该位置已经访问 //获取当前位置可以走的下一个位置的集合 ArrayList<Point> ps = next(new Point(column, row)); //对ps进行排序,排序的规则就是对ps的所有的Point对象的下一步的位置的数目,进行非递减排序 sort(ps); //遍历 ps while(!ps.isEmpty()) { Point p = ps.remove(0);//取出下一个可以走的位置 //判断该点是否已经访问过 if(!visited[p.y * X + p.x]) {//说明还没有访问过 traversalChessboard(chessboard, p.y, p.x, step + 1); } } //判断马儿是否完成了任务,使用 step 和应该走的步数比较 //如果没有达到数量,则表示没有完成任务,将整个棋盘置0 //说明: step < X * Y 成立的情况有两种 //1. 棋盘到目前位置,仍然没有走完 //2. 棋盘处于一个回溯过程 if(step < X * Y && !finished ) { chessboard[row][column] = 0; visited[row * X + column] = false; } else { finished = true; } } //根据当前位置(Point对象),计算马儿还能走哪些位置(Point),并放入到一个集合中(ArrayList),最多有8个位置 public static ArrayList<Point> next(Point curPoint) { //创建一个ArrayList ArrayList<Point> ps = new ArrayList<Point>(); //创建一个Point Point p1 = new Point(); //表示马儿可以走5这个位置 if((p1.x = curPoint.x - 2) >= 0 && (p1.y = curPoint.y -1) >= 0) { ps.add(new Point(p1)); } //判断马儿可以走6这个位置 if((p1.x = curPoint.x - 1) >=0 && (p1.y=curPoint.y-2)>=0) { ps.add(new Point(p1)); } //判断马儿可以走7这个位置 if ((p1.x = curPoint.x + 1) < X && (p1.y = curPoint.y - 2) >= 0) { ps.add(new Point(p1)); } //判断马儿可以走0这个位置 if ((p1.x = curPoint.x + 2) < X && (p1.y = curPoint.y - 1) >= 0) { ps.add(new Point(p1)); } //判断马儿可以走1这个位置 if ((p1.x = curPoint.x + 2) < X && (p1.y = curPoint.y + 1) < Y) { ps.add(new Point(p1)); } //判断马儿可以走2这个位置 if ((p1.x = curPoint.x + 1) < X && (p1.y = curPoint.y + 2) < Y) { ps.add(new Point(p1)); } //判断马儿可以走3这个位置 if ((p1.x = curPoint.x - 1) >= 0 && (p1.y = curPoint.y + 2) < Y) { ps.add(new Point(p1)); } //判断马儿可以走4这个位置 if ((p1.x = curPoint.x - 2) >= 0 && (p1.y = curPoint.y + 1) < Y) { ps.add(new Point(p1)); } return ps; } //根据当前这个一步的所有的下一步的选择位置,进行非递减排序,减少回溯的次数 public static void sort(ArrayList<Point> ps) { ps.sort(new Comparator<Point>() { @Override public int compare(Point o1, Point o2) { //获取到o1的下一步的所有位置个数 int count1 = next(o1).size(); //获取到o2的下一步的所有位置个数 int count2 = next(o2).size(); if(count1 < count2) { return -1; } else if (count1 == count2) { return 0; } else { return 1; } } }); } }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。