Java实现插入排序、冒泡排序、堆排序、希尔排序、选择排序、优先队列排序、快速排序、归并排序(详细注释,原理解析)

插入排序

package learn;import java.util.Arrays;/** 每次都将当前元素插入到左侧已经排序的数组中,使得插入之后左侧数组依然有序。* 速度优于选择排序*/
public class InsertSort {public static void insertSort(int[] a) {int n = a.length;for (int i = 1; i < n; i++) {// 为了将当前元素插入到已经排序的数组中需要从后往前遍历for (int j = i; j > 0 ; j--) {if (a[j] < a[j - 1]) {int temp = a[j];a[j] = a[j - 1];a[j - 1] = temp;} else {continue;}}}}public static void main(String[] args) {
//        int[] s = {4, 6, 1, 2, 3, 0};int[] s = {9,7,1,-8,36,-18,100};System.out.println(Arrays.toString(s));insertSort(s);System.out.println(Arrays.toString(s));}
}

package sort;/*** 【大顶堆】* 堆中某个节点的值总是大于等于其子节点的值,并且堆是一颗完全二叉树。** 堆可以用数组来表示,这是因为堆是完全二叉树,而完全二叉树很容易就存储在数组中。* 位置 k 的节点的父节点位置为 k/2,而它的两个子节点的位置分别为 2k 和 2k+1。* @param <T>*/
public class dui<T extends Comparable<T>> {private T[] heap;private int N = 0;public dui(int maxN) {this.heap = (T[]) new Comparable[maxN + 1];}public boolean isEmpty() {return N == 0;}public int size() {return N;}private boolean less(int i, int j) {return heap[i].compareTo(heap[j]) < 0;}private void swap(int i, int j) {T t = heap[i];heap[i] = heap[j];heap[j] = t;}// 上浮:把大元素向上排private void swim(int k) {// k 的父节点 k/2 更小的时候需要对 k 做上浮操作while (k > 1 && less(k / 2, k)) {swap(k / 2, k);// 交换k = k / 2;// 更改当前 k 的位置}}// 下沉:把小元素往下排private void sink(int k) {// k 的子节点分别是 k*2 与 k*2+1while (2 * k <= N) {int j = 2 * k;if (j < N && less(j, j + 1))// 左子节点要小于右子节点j++;// 若父节点 >= 左子节点 就表示if (!less(k, j))break;// 父节点 < 左子节点,就需要交换swap(k, j);k = j;}}// 插入元素:将新元素放到数组末尾,然后上浮到合适的位置。public void insert(T v) {heap[++N] = v;swim(N);}// 删除最大元素:// 从数组顶端删除最大的元素,并将数组的最后一个元素放到顶端,并让这个元素下沉到合适的位置。public T delMax() {T max = heap[1];swap(1, N--);heap[N + 1] = null;sink(1);return max;}
}

堆排序

package sort;import java.util.Arrays;/*** 把最大元素和当前堆中数组的最后一个元素交换位置,并且不删除它,* 那么就可以得到一个从尾到头的递减序列,从正向来看就是一个递增序列,* 这就是堆排序。*/
public class HeapSort {public static void heapSort(int[] a) {int N = a.length - 1;// 从右至左进行下沉操作for (int k = N / 2; k >= 1; k--) {sink(a, k, N);}while (N > 1) {swap(a, 1, N--);sink(a, 1, N);}}// 下沉private static void sink(int[] a, int k, int N) {while (2 * k <= N) {int j = 2 * k;if (j < N && less(a[j], a[j + 1]))j++;if (!less(a[k], a[j]))break;swap(a, k, j);k = j;}}private static boolean less(Comparable v, Comparable w) {// 若 v 小于 w 则返回负数return v.compareTo(w) < 0;}private static void swap(int[] a, int i, int j) {int t = a[i];a[i] = a[j];a[j] = t;}public static void main(String[] args) {int[] a = {1, 9, 2, 8, 3, 6, 7};System.out.println(Arrays.toString(a));heapSort(a);System.out.println(Arrays.toString(a));}}

快速排序

package sort;import java.util.Arrays;/*** 使用最广泛的排序算法* 原理上是一种分治算法,将一个数组分为两部分,对这两部分独立的排序,然后整体有序* 与归并排序的区别:* 1):归并排序虽然也是把数组分为两部分,但是归并排序需要在部分有序后再做一次排序来让整体有序;* 但是快速排序是:部分有序了,整体自然就有序了,无需对整体再一次排序。* 2):归并排序的数组切分是平分,但是快速排序的数组切分是根据数组内容由函数产生的*/
public class QuickSort {public static void quickSort(int[] a, int start, int end) {if (end <= start) {return;}// j 是中分点,parttion算法是关键int j = partition(a, start, end);quickSort(a, start, j - 1);// 针对左边排序quickSort(a, j + 1, end); // 针对右边排序}private static boolean less(Comparable v, Comparable w) {// 若 v 小于 w 则返回负数return v.compareTo(w) < 0;}private static void exch(int[] a, int i, int j) {int t = a[i];a[i] = a[j];a[j] = t;}/*这段代码按照a[start]的值v进行切分。当指针i和j相遇时主循环退出。在循环中,a[i]小于v时我们增大i,a[j]大于v时我们减小j,然后交换a[i]和a[j]来保证i左侧的元素都不大于v, j右侧的元素都不小于V。当指针相遇时交换a[start]和a[j],切分结束(这样切分值就留在a[j]中了)。*/private static int partition(int[] a, int start, int end) {// i、j 两根指针分别从左右开始寻找第一个大于 v 的与第一个小于 v 的值int i = start;int j = end + 1;int v = a[start];while (true) {// 找到第一个大于 v 的值与 iwhile (less(a[++i], v)) {if (i == end) {break;}}// 找到第一个小于 v 的值与 jwhile (less(v, a[--j])) {if (j == start) {break;}}// i、j 指针相交后if (i >= j) {break;}// 交换 i、j 处的值,确保 i 左边的都不大于 v,j 右边的都不小于 vexch(a, i, j);}exch(a, start, j);// 将 V = a[j]放入正确的位置return j;}public static void main(String[] args) {
//        int[] a = {12, 6, 9, 3, 78, 3, 3, 4};int[] a = {9, 7, 1, -8, -1, -18, 100};System.out.println(Arrays.toString(a));quickSort(a, 0, a.length - 1);System.out.println(Arrays.toString(a));}
}

冒泡排序

package sort;public class BubbleSort {public static void bubbleSort(int[] a) {// 冒泡排序的最大的特点就是顺序是从右往前逐渐有序的// 所以,随着排序的逐渐执行,右边的越来越有序,但是每一次比较缺还在比较已经有序的部分// 因此需要排除遍历这部分有序的数据,也就是内循环只遍历到右边有序数据的前面就行了,不能再往后遍历了int sortedBorder = a.length - 1;// 这个 lastSwapIndex 肯定是从大数开始减少的,代表着右边的数据在排序int lastSwapIndex = 0;for (int i = 0; i < a.length; i++) {boolean isSorted = true;for (int j = 0; j < sortedBorder; j++) {if (a[j] > a[j + 1]) {a[j] ^= a[j + 1];a[j + 1] ^= a[j];a[j] ^= a[j + 1];isSorted = false;lastSwapIndex = j;}}// 内循环遍历结束,代表着,右边已经部分有序了,最后一次排序的 j 就是有序无序的边界sortedBorder = lastSwapIndex;if (isSorted) {break;}}}public static void main(String[] args) {int[] arr = {6, 3, 8, 2, 9, 1, 0, 5, 4, 7, -1};
//        int[] arr = {6, 3, 8, 12, 19, 110};
//        int[] arr = {0, 3, 8, 12, 19, 110};System.out.println("排序前数组为:");for (int num : arr) {System.out.print(num + " ");}bubbleSort(arr);System.out.println();System.out.println("排序后的数组为:");for (int num : arr) {System.out.print(num + " ");}}
}

希尔排序

package sort;import java.util.Arrays;
/*
希尔排序是基于插入排序的一种快速排序算法
插入排序的缺点:对于大规模的乱序数组排序速度慢
希尔排序简单的改变了插入排序:交换不相邻的元素以对数组的局部进行排序,
并最终用插入排序将局部有序的数组排序希尔排序对于大型数组排序效率很高,不需要额外的内存空间*/
public class ShellSort {public static void shellSort(int[] a) {int n = a.length;// 设置增量,增量的取法有很多,这里是推荐取法// 插入排序的增量是1,属于相邻元素比较,现在换成不相邻元素比较int h = 1;while (h < n / 3) {h = 3 * h + 1;}// 增量最小为 1 ,也就是相邻的两个元素比较while (h >= 1) {// 对相聚 h 的两个元素使用插入排序for (int i = h; i < n; i++) {for (int j = i; j >= h && less(a[j], a[j-h]); j -= h) {exch(a, j, j-h);}}// 排序结束后,缩小增量,继续排序h /= 3;}}private static void exch(int[] a, int i, int j) {int t = a[i];a[i] = a[j];a[j] = t;}private static boolean less(Comparable v, Comparable w) {// 若 v 小于 w 则返回负数return v.compareTo(w) < 0;}public static void main(String[] args) {
//        int[] s = {3, -2, -1};int[] s = {9,7,1,-8,-1,-18,100};System.out.println(Arrays.toString(s));shellSort(s);System.out.println(Arrays.toString(s));}
}

选择排序

package sort;import java.util.Arrays;/*** 在数组里面选择最小的,放到第一位,然后在剩下的数组里面找到最小的放到第二位,如此反复直到结束*/
public class SelectSort {public static void selectSort(int[] a) {int n = a.length;for (int i = 0; i < n-1; i++) {for (int j = i + 1; j < n; j++) {// 要始终保持最小值为当前的 iint minIndex = i;boolean exchange = true;// 如果 a[j] < a[min] 说明需要把 j 设置为最小的if (a[j] < a[minIndex]) {minIndex = j;} else {exchange = false;}// 如果需要交换数据就开始交换if (exchange) {int temp = a[i];a[i] = a[j];a[j] = temp;}}}}public static void main(String[] args) {int[] array = {0, 200, -523, 7, 1, -1, -18, 2, 10, -6};System.out.println(Arrays.toString(array));selectSort2(array);System.out.println(Arrays.toString(array));}public static void selectSort2(int[] array) {for (int i = 0; i < array.length; i++) {for (int j = i + 1; j < array.length; j++) {// 假设 i 所处位置的元素就是最小的int min = i;if (array[j] < array[min]) {// 后面的更小就要排序到前面array[i] ^= array[j];array[j] ^= array[i];array[i] ^= array[j];}}}}
}

优先队列排序

package sort;import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.Queue;public class Test {private static class Customer {private Integer id;private String name;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public Customer(Integer id, String name) {this.id = id;this.name = name;}public Customer() {}@Overridepublic String toString() {return  "weight=" + id +", name=" + name;}}public static void main(String[] args) {// 1.演示优先队列的自然排序Queue<Integer> naturePriorityQueue = new PriorityQueue<>(10);for (int i = 0; i < 10; i++) {int random = (int) (Math.random() * 100000 + 1);System.out.println("add i value = " + random);naturePriorityQueue.add(random);Integer peek = naturePriorityQueue.peek();System.out.println("poll i value = " + peek);System.out.println();}// 2.演示自定义排序Queue<Customer> customerPriorityQueue = new PriorityQueue<>(10, idComparator);// 初始化队列大小并指定比较器for (int i = 0; i < 10; i++) {Customer customer = new Customer((int) (Math.random() * 6532 + 1), "customer" + i);System.out.println("add customer is " + customer.toString());customerPriorityQueue.add(customer);Customer customer1 = customerPriorityQueue.peek();System.out.println("peek customer is " + customer1.toString());System.out.println();}}// 自定义的比较器public static Comparator<Customer> idComparator = new Comparator<Customer>() {@Overridepublic int compare(Customer o1, Customer o2) {return o1.getId() - o2.getId();}};}

自顶向下的归并实现

package sort;import java.util.Arrays;public class MergeSortUp2Down {/*该方法用来拆解原数组为一个个单一的元素*/public static void mergeSortUp2Down(int[] array, int start, int end) {if (start >= end)return;int mid = (start + end) >> 1; // 把除法换成位运算mergeSortUp2Down(array, start, mid);mergeSortUp2Down(array, mid + 1, end);// 拆完后进入比较排序处理阶段merge(array, start, mid, end);}/*每一次进入该方法就代表一个小阶段的归并需要一个临时数组存储排序的数据(不是修改原始数据)归并完了之后可能还有一部分数据本来有序,无需归并,把他们加入到临时数组后面(到此为止归并才真正的结束)(现在知道归并时为什么不能修改原始数组了吧)最后把归并的结果写回到原始数组中*/private static void merge(int[] array, int start, int mid, int end) {// 归并的特点是两两比对,但不是只比对两个数,是两组!用 i、j 指向这两组元素的第一个int i = start;int j = mid + 1;int index = 0;// 存储新排序的数据int[] tmp = new int[end - start + 1];while (i <= mid && j <= end) { // 前面的一组不能越 mid 界,后面一组不可以越 end 界if (array[j] <= array[i]) {tmp[index++] = array[j++];} else {tmp[index++] = array[i++];}}while (i <= mid) {tmp[index++] = array[i++];}while (j <= end) {tmp[index++] = array[j++];}// 把现在的有序数据写回到原始数组里面for (index = 0; index < tmp.length; index++) {array[start + index] = tmp[index];}}public static void main(String[] args) {int[] a = {6, 3, 7, 2, 2, 3, 4, 9, 1, -1};mergeSortUp2Down(a, 0, a.length - 1);System.out.println(Arrays.toString(a));}
}

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

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

相关文章

Elasticsearch优点和缺点以及要点和难点具体应用

Elasticsearch是一个开源、分布式、实时的搜索和分析引擎,它位于Elastic Stack(以前称为ELK Stack)的核心。以下是关于Elasticsearch的一些主要特点和功能: 1.分布式和可扩展性:Elasticsearch是分布式的,可以轻松扩展到多个节点以处理大规模数据集和高并发请求。通过将数…

C语言 | Leetcode C语言题解之第113题路径总和II

题目&#xff1a; 题解&#xff1a; int** ret; int retSize; int* retColSize;int* path; int pathSize;typedef struct {struct TreeNode* key;struct TreeNode* val;UT_hash_handle hh; } hashTable;hashTable* parent;void insertHashTable(struct TreeNode* x, struct Tr…

C++干货 --类和对象(二)

前言&#xff1a; 上文中&#xff0c;我们介绍了类这一重要知识点&#xff0c;包括为什么要有类、类的使用方法、封装、以及对象实例化。详情可以去看我的文章&#xff1a;写文章-CSDN创作中心C干货 --类和对象(一)-CSDN博客写文章-CSDN创作中心 这篇文章&#xff0c;我们简单…

我学会了用插件来辅助PostgreSQL,可拷,很刑!

作者&#xff1a;IT邦德 中国DBA联盟(ACDU)成员&#xff0c;10余年DBA工作经验&#xff0c; Oracle、PostgreSQL ACE CSDN博客专家及B站知名UP主&#xff0c;全网粉丝10万 擅长主流Oracle、MySQL、PG、高斯及Greenplum备份恢复&#xff0c; 安装迁移&#xff0c;性能优化、故障…

【蓝桥杯省赛真题44】python计算N+N的值 中小学青少年组蓝桥杯比赛 算法思维python编程省赛真题解析

目录 python计算NN的值 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 七、 推荐资料 1、蓝桥杯比赛 2、考级资料 3、其它资料 python计算NN的值 第十四届蓝桥杯青少年组python省赛真题 一、题目要求…

Makefile学习笔记18|u-boot顶层Makefile04

Makefile学习笔记18|u-boot顶层Makefile04 希望看到这篇文章的朋友能在评论区留下宝贵的建议来让我们共同成长&#xff0c;谢谢。 这里是目录 C语言标准 # With the move to GCC 6, we have implicitly upgraded our language # standard to GNU11 (see https://gcc.gnu.org/…

WordPress--解决迁移后页面404(打不开)的问题

原文网址&#xff1a;WordPress--解决迁移后页面404的问题_IT利刃出鞘的博客-CSDN博客 简介 本文介绍WordPress迁移后部分页面404(打不开)的解决方案。 方案&#xff1a;修改固定链接 简介 要在刚建好网站&#xff0c;还没导入数据之前修改固定链接&#xff0c;否则可能导…

【教学类-58-05】黑白三角拼图05(2-10宫格,每个宫格随机1张-6张,带空格纸,1页3张黑白3张白卡)

背景需求&#xff1a; 【教学类-58-04】黑白三角拼图04&#xff08;2-10宫格&#xff0c;每个宫格随机1张-6张&#xff0c;带空格纸&#xff0c;1页6张黑白&#xff0c;1张6张白卡&#xff09;-CSDN博客文章浏览阅读582次&#xff0c;点赞16次&#xff0c;收藏3次。【教学类-58…

Mac安装pytorch

先下载 Anaconda | The Operating System for AI 网速慢&#xff0c;用中国大陆镜像&#xff1a;NJU Mirror 之前装python3时用的是pip3&#xff0c;这里说一下这pip与conda的区别 Conda和pip都是Python包管理工具&#xff0c;用于安装和管理Python包 包管理范围&#xff1a…

汽车数据应用构想(一)

自从电动汽车GB/T32960标准颁布&#xff0c;要求所有电动汽车必须上传数据开始&#xff0c;各车厂就开始花费大量的人力物力&#xff0c;用于数据的上传与存储。同时随着智能化、网联化的趋势&#xff0c;不断丰富上传数据的内容与数量。数据已成为车厂的重要资产&#xff0c;但…

Day16

Day16 一、迭代器 深入迭代器-foreach的底层 for (String element : list) {System.out.println(element);}底层&#xff1a; //使用foreach循环遍历集合的底层实现&#xff1a;String element;for(Iterator it list.iterator();it.hasNext();System.out.println(element))e…

微服务:eureka的搭建,以及服务注册、服务发现、负载均衡

eureka 搭建 新建一个Module,maven项目&#xff0c;导入依赖。 <dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-server</artifactId></dependency><…

月入10万+管道收益,揭秘旅游卡运营的5个阶段!

网上的项目确实繁多&#xff0c;只要深入研究&#xff0c;总能找到适合自己的赚钱方式。在互联网上运营&#xff0c;关键在于理解其底层逻辑。旅游卡项目&#xff0c;就是一个能实现月入10万的实例。接下来&#xff0c;我将为你揭密旅游卡运营的5个阶段&#xff1a; 1、先赚成…

24年湖南教资认定即将开始,别被照片卡审!

24年湖南教资认定即将开始&#xff0c;别被照片卡审&#xff01;

Jenkins 流水线(Pipeline)详解

大家好&#xff0c;Jenkins 流水线&#xff08;Pipeline&#xff09;是一种可编排的持续集成和交付&#xff08;CI/CD&#xff09;方法&#xff0c;它以代码的方式定义整个软件开发过程中的构建、测试和部署流程。接下来就跟大家分享一下Jenkins 流水线&#xff08;Pipeline&am…

2024年5月26日 (周日) 叶子游戏新闻

资深开发者&#xff1a;3A游戏当前处于一种尴尬的中间地带游戏行业整体&#xff0c;尤其是3A游戏正处于艰难时期。尽管2023年3A游戏佳作频出&#xff0c;广受好评&#xff0c;但居高不下的游戏开发成本&#xff08;传闻《漫威蜘蛛侠2》的制作成本高达3亿美元&#xff09;正严重…

【机器学习结合AI绘画工具】——开启艺术创作的新纪元

目录 一、AI绘画工具的发展历程 二、AI绘画工具的技术原理 实例说明 三、AI绘画工具在艺术创作中的应用 实例网站 四、AI绘画工具的影响与未来展望 结论 机器学习和人工智能&#xff08;AI&#xff09;在过去的十年里取得了显著的进展。特别是在艺术创作领域&#xff0c…

【笔记】太久没有写selenium了

以前qyqt和selenium用来开发一个自动化测试工具&#xff0c;现在回忆一下已经过了将近6年。 DeprecationWarning: find_element_by_* commands are deprecated. Please use find_element() 例如&#xff1a;find_element_by_xpath改为web.find_element(By.XPATH from seleniu…

excel表格写存神器--xlwt

原文链接&#xff1a;http://www.juzicode.com/python-tutorial-xlwt-excel 在 Python进阶教程m2d–xlrd读excel 中我们介绍了Excel表格的读取模块xlrd&#xff0c;今天这篇文章带大家了解Excel表格写存模块xlwt。他俩名字相近都以Excel的简写xl开头&#xff0c;rd是read的简写…

Java顺序表

Java顺序表 前言一、线性表介绍常见线性表总结图解 二、顺序表概念顺序表的分类顺序表的实现throw具体代码 三、顺序表会出现的问题 前言 推荐一个网站给想要了解或者学习人工智能知识的读者&#xff0c;这个网站里内容讲解通俗易懂且风趣幽默&#xff0c;对我帮助很大。我想与…