图解算法--排序算法

目录

1.冒泡排序算法

2.选择排序算法

3.插入排序算法

4.希尔排序算法

5.归并排序算法

6.快速排序算法


1.冒泡排序算法

原理讲解:

  1. 从待排序的数组中的第一个元素开始,依次比较当前元素和它相邻的下一个元素的大小。
  2. 如果当前元素大于相邻元素,则交换这两个元素的位置,将较大的元素向后冒泡。
  3. 继续比较相邻元素,重复以上步骤,直到遍历完整个数组。
  4. 一轮遍历完成后,最大的元素将会排在最后面。
  5. 重复执行上述步骤,每次遍历都将会使待排序的元素中最大的元素冒泡到正确的位置,直到所有元素都有序排列。

 传统的冒泡排序算法

package Sort.Bubble;import java.util.Arrays;public class javaDemo {public static void main(String[] args) {int nums[] = new int[]{1,4,36,7,42,2,12,5,2};int temp;
//        冒泡算法for (int i=0;i< nums.length-1;i++){for (int j=0;j< nums.length-i-1;j++){if (nums[j]>nums[j+1]){temp = nums[j];nums[j] = nums[j+1];nums[j+1] = temp;}}}System.out.println(Arrays.toString(nums));}
}

传统的排序算法有个缺点,那就是不管数据是否已经完成排序都会固定执行n(n-1)/2次,而如果加入岗哨的概念进行改造冒泡排序算法就可以提高程序的执行效率

传统冒泡执行已经快要排序好的数组结果如下:

可以看到在第二轮时候就已经排序好了,但是还是不断进行遍历,所以加入一个岗哨(flag)判断功能,当第三轮时候,如果没有元素进行交换后直接退出排序 。

改进型冒泡排序法

package Sort.Bubble;import java.util.Arrays;public class Sentry {public static void main(String[] args) {int num[] = new int[]{1,4,5,2,7,8,9};int flag;int temp;for (int i= num.length-1;i>0;i--){flag = 0;
//            如果发生了交换则不跳出for (int j=0;j<i-1;j++){if (num[j]>num[j+1]){temp = num[j];num[j] = num[j+1];num[j+1] = temp;flag ++;}}
//            如果从头遍历结束都没有交换则直接退出if (flag == 0){break;}System.out.println("第"+(num.length-i)+"轮排序结果为"+ Arrays.toString(num));}}
}

 算法分析:

  • 冒泡排序算法在最坏的情况下时间复杂度是O(n^2)
  • 在最好的情况下时间复杂度是O(n)

2.选择排序算法

原理讲解: 

  1. 遍历待排序的数组,将第一个元素作为当前最小值。
  2. 从当前位置之后的元素中找到最小的元素,并记录其位置。
  3. 如果找到了比当前最小值更小的元素,则将该元素的位置更新为当前最小值的位置。
  4. 遍历完整个数组后,将最小值与当前位置的元素进行交换。
  5. 继续从下一个位置开始重复以上步骤,直到遍历完整个数组。

案例代码

package Sort.Chose.chose;import java.util.Arrays;public class javaDemo {public static void main(String[] args) {int nums[] = new int[]{1,2,4,2,3,51,12,6,2};int min;int index;int temp;for (int i=0;i< nums.length-1;i++){
//            每一轮初始化最小值与最小值下角标min = nums[i];index = i;for (int j=i+1;j< nums.length;j++){if (min>nums[j]){min = nums[j];index = j;}}temp = nums[i];nums[i] = min;nums[index] = temp;}System.out.println(Arrays.toString(nums));}
}

 算法分析:

  • 冒泡排序算法无论在最坏还是最好的情况下时间复杂度都是O(n^2)

3.插入排序算法

 原理讲解:

  1. 将数组分为已排序和未排序两部分。一开始,将第一个元素视为已排序部分,其余元素视为未排序部分。
  2. 从未排序部分选择第一个元素,依次与已排序部分的元素比较。
  3. 如果选取的元素小于已排序部分的某个元素,则将该元素插入到正确的位置,同时将已排序部分中的元素向后移动一位。
  4. 重复上述步骤,将未排序部分的每个元素逐个插入到已排序部分的正确位置。
  5. 当所有元素都插入完成,即未排序部分为空时,排序完成。

案例代码  

package Sort.InsertSort;import java.util.Arrays;public class javaDemo {public static void main(String[] args) {int nums[] = new int[]{4,1,5,2,6,7,2};int temp;for (int i=0;i< nums.length;i++){for (int j=0;j<i;j++){if (nums[i]<nums[j]){temp = nums[j];nums[j] = nums[i];nums[i] = temp;}}System.out.println("第"+(i+1)+"轮排序结果为"+ Arrays.toString(nums));}}
}

算法分析:

  • 冒泡排序算法在最坏的情况下时间复杂度是O(n^2)
  • 在最好的情况下时间复杂度是O(n)

4.希尔排序算法

  原理讲解:

  1. 首先,选择一个增量 gap,通常将数组长度的一半作为初始增量。
  2. 将数组按照增量进行分组,并对每个分组使用插入排序算法进行排序。
  3. 逐渐缩小增量,重复以上步骤,直到增量为 1。
  4. 最后使用增量为 1 的插入排序对整个数组进行最后一次排序

案例代码:

package Sort.Shell;import java.util.Arrays;public class ShellSort {public static void main(String[] args) {int[] arr = {9, 5, 2, 7, 1, 8, 3, 6, 4};int n = arr.length;// 定义增量序列int[] increments = {5, 3, 1};System.out.println("原始数组顺序为"+Arrays.toString(arr));// 遍历增量序列for (int increment : increments) {// 对每个增量进行插入排序for (int i = increment; i < n; i++) {int temp = arr[i];int j = i;// 在当前增量下,对子序列进行插入排序while (j >= increment && arr[j - increment] > temp) {arr[j] = arr[j - increment];j -= increment;}arr[j] = temp;System.out.println("当前增量为"+increment+",当前数组为:"+Arrays.toString(arr));}}}
}

算法分析:

  • 希尔排序算法无论在最坏还是最好的情况下时间复杂度都是O(n^3/2)
  • 空间最优解

5.归并排序算法

   原理讲解:

  1. 将待排序的数组递归地分成两个子数组,直到每个子数组只有一个元素。
  2. 对每个子数组进行合并操作,将两个有序的子数组合并成一个有序的数组。
  3. 重复以上步骤,直到所有的子数组都被合并成一个完整的有序数组。

案例代码:

package Sort.MergeSort;import java.util.Arrays;public class mergeSort {public static void mergeSort(int[] arr) {if (arr == null || arr.length <= 1) {return;}int[] temp = new int[arr.length];mergeSort(arr, 0, arr.length - 1, temp);}private static void mergeSort(int[] arr, int left, int right, int[] temp) {if (left < right) {int mid = left + (right - left) / 2;mergeSort(arr, left, mid, temp); // 递归排序左半部分mergeSort(arr, mid + 1, right, temp); // 递归排序右半部分merge(arr, left, mid, right, temp); // 合并左右两个有序数组}System.out.println("当前顺序为"+Arrays.toString(arr));}private static void merge(int[] arr, int left, int mid, int right, int[] temp) {int i = left; // 左半部分起始索引int j = mid + 1; // 右半部分起始索引int k = 0; // 临时数组索引// 将左右两个有序数组按顺序合并到temp数组中while (i <= mid && j <= right) {if (arr[i] <= arr[j]) {temp[k++] = arr[i++];} else {temp[k++] = arr[j++];}}// 处理剩余的元素while (i <= mid) {temp[k++] = arr[i++];}while (j <= right) {temp[k++] = arr[j++];}// 将临时数组的元素拷贝回原数组for (i = 0; i < k; i++) {arr[left + i] = temp[i];}}public static void main(String[] args) {int[] arr = {9, 5, 2, 7, 1, 8, 3, 6, 4};System.out.println("原始数组:"+ Arrays.toString(arr));mergeSort(arr);}
}

算法分析:

  • 归并排序算法无论在最坏还是最好的情况下时间复杂度都是O(nlogn)

6.快速排序算法

   原理讲解:

  1. 选择一个基准元素(通常是数组的第一个或最后一个元素)作为参考点。
  2. 将待排序的数组分成两个子数组,其中一个子数组中的元素小于等于基准元素,另一个子数组中的元素大于基准元素。
  3. 对这两个子数组分别重复上述步骤,递归地进行快速排序。
  4. 当子数组的长度为 1 或 0 时,递归终止。
  5. 将排序好的子数组合并起来,即可得到完整的有序数组。

案例代码:

package Sort.Quick;import java.util.Arrays;public class QuickSort {public static void quickSort(int[] arr) {if (arr == null || arr.length <= 1) {return;}quickSort(arr, 0, arr.length - 1);}private static void quickSort(int[] arr, int low, int high) {if (low < high) {int pivotIndex = partition(arr, low, high); // 将数组划分为两部分,返回中轴元素的索引quickSort(arr, low, pivotIndex - 1); // 递归排序左半部分quickSort(arr, pivotIndex + 1, high); // 递归排序右半部分}System.out.println("当前数组为"+Arrays.toString(arr));}private static int partition(int[] arr, int low, int high) {int pivot = arr[low]; // 选择第一个元素作为中轴元素int left = low;int right = high;while (left < right) {// 从右向左找第一个小于等于中轴元素的位置while (left < right && arr[right] >= pivot) {right--;}if (left < right) {arr[left] = arr[right];left++;}// 从左向右找第一个大于中轴元素的位置while (left < right && arr[left] <= pivot) {left++;}if (left < right) {arr[right] = arr[left];right--;}}arr[left] = pivot; // 将中轴元素放置到最终位置return left; // 返回中轴元素的索引}public static void main(String[] args) {int[] arr = {9, 5, 2, 7, 1, 8, 3, 6, 4};System.out.printf("原始数组:");System.out.println(Arrays.toString(arr));quickSort(arr);}
}

 算法分析:

  • 快速排序算法在最坏的情况下时间复杂度是O(nlog2n)
  • 在最好的情况下时间复杂度是O(n)
  • 公认最好的排序算法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/72742.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【JavaEE进阶】MyBatis的创建及使用

文章目录 一. MyBatis简介二. MyBatis 使用1. 数据库和数据表的创建2. 创建Mybatis项目2.1 添加MyBatis框架支持2.2 设置MyBatis配置信息 3. MyBatis开发流程4. MyBatis查询数据库测试 三. MyBatis 流程1. MyBatis 查询数据库流程2. MyBatis 框架交互流程图 一. MyBatis简介 M…

《Java极简设计模式》第04章:建造者模式(Builder)

作者&#xff1a;冰河 星球&#xff1a;http://m6z.cn/6aeFbs 博客&#xff1a;https://binghe.gitcode.host 文章汇总&#xff1a;https://binghe.gitcode.host/md/all/all.html 源码地址&#xff1a;https://github.com/binghe001/java-simple-design-patterns/tree/master/j…

什么是微服务?

2.微服务的优缺点 优点 单一职责原则每个服务足够内聚&#xff0c;足够小&#xff0c;代码容易理解&#xff0c;这样能聚焦一个指定的业务功能或业务需求&#xff1b;开发简单&#xff0c;开发效率提高&#xff0c;一个服务可能就是专一的只干一件事&#xff1b;微服务能够被小…

TCP中窗口和滑动窗口的含义以及流量控制

一.窗口 在TCP中由于要保证可靠性&#xff0c;所以每发送一条数据后&#xff0c;都需要接收方返回一条应答报文&#xff0c;要是我们每发送一条数据&#xff0c;发送方就等待接收应答报文&#xff0c;收到之后再去发送下一条数据&#xff0c;这样我们就会花费大量的时间在等待应…

Python土力学与基础工程计算.PDF-钻探泥浆制备

Python 求解代码如下&#xff1a; 1. rho1 2.5 # 黏土密度&#xff0c;单位&#xff1a;t/m 2. rho2 1.0 # 泥浆密度&#xff0c;单位&#xff1a;t/m 3. rho3 1.0 # 水的密度&#xff0c;单位&#xff1a;t/m 4. V 1.0 # 泥浆容积&#xff0c;单位&#xff1a;…

Streamlit 讲解专栏(十):数据可视化-图表绘制详解(上)

文章目录 1 前言2 st.line_chart&#xff1a;绘制线状图3 st.area_chart&#xff1a;绘制面积图4 st.bar_chart&#xff1a;绘制柱状图5 st.pyplot&#xff1a;绘制自定义图表6 结语 1 前言 在数据可视化的世界中&#xff0c;绘制清晰、易于理解的图表是非常关键的。Streamlit…

1000元到3000元预算的阿里云服务器配置大全

1000元、2000元或3000元预算能够买的阿里云服务器配置有哪些&#xff1f;可以选择ECS通用算力型u1云服务器、ECS计算型c7或通用型g7实例规格&#xff0c;当然&#xff0c;如果选择轻量应用服务器的话&#xff0c;更省钱&#xff0c;阿里云百科分享1000-3000元预算能买的阿里云服…

谷歌云 | 最大限度地提高可靠性降低成本:调整 Kubernetes 工作负载的大小

【Cloud Ace 是谷歌云全球战略合作伙伴&#xff0c;拥有 300 多名工程师&#xff0c;也是谷歌最高级别合作伙伴&#xff0c;多次获得 Google Cloud 合作伙伴奖。作为谷歌托管服务商&#xff0c;我们提供谷歌云、谷歌地图、谷歌办公套件、谷歌云认证培训服务。】 您知道通过调整…

5.利用matlab完成 符号矩阵的转置和 符号方阵的幂运算(matlab程序)

1.简述 Matlab符号运算中的矩阵转置 转置向量或矩阵 B A. B transpose(A) 说明 B A. 返回 A 的非共轭转置&#xff0c;即每个元素的行和列索引都会互换。如果 A 包含复数元素&#xff0c;则 A. 不会影响虚部符号。例如&#xff0c;如果 A(3,2) 是 12i 且 B A.&#xff0…

grafana中利用变量来添加dashboard详情页地址实现点击跳转

背景 最近弄grafana的dashboard,突然想到各个dashboard之前可以直接跳转到不同详细页面的面板,于是找了找实现方法 实现 以stat 格式的面板为例,显示出各个pod的对应状态, PromQL是(avg(kube_pod_status_phase{phase"Running", namespace!"kube-system"…

什么是异常处理

文章目录 异常处理介绍自定义异常页面文档:自定义异常页面说明 自定义异常页面-应用实例需求:代码实现 全局异常说明全局异常-应用实例需求:代码实现完成测试 自定义异常说明自定义异常-应用实例需求&#xff1a;代码实现完成测试 注意事项完成测试 异常处理 介绍 默认情况下…

【路由协议】使用按需路由协议和数据包注入的即时网络模拟传递率(PDR)、总消耗能量和节点消耗能量以及延迟研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…