归并排序的原理及java代码实现
作者:hebedich 时间:2021-11-18 13:51:10
概述
归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。
归并排序采用的是递归来实现,属于“分而治之”,将目标数组从中间一分为二,之后分别对这两个数组进行排序,排序完毕之后再将排好序的两个数组“归并”到一起,归并排序最重要的也就是这个“归并”的过程,归并的过程中需要额外的跟需要归并的两个数组长度一致的空间。
效果图:
步骤
申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列
设定两个指针,最初位置分别为两个已经排序序列的起始位置
比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置
重复步骤3直到某一指针达到序列尾
将另一序列剩下的所有元素直接复制到合并序列尾
实例
原始数据:
3 5 2 6 2
归并的前提是将数组分开,一分为二,再一分为二,分到不能再分,进行归并。
第一轮分隔,索引2 ((0+4)/2=2) 为中间
[3 5 2] [6 2]
第二轮分隔,对[3 5 2]进行分隔
[3 5] [2] [6 2]
第三轮分隔,对[3 5]进行分隔
[3] [5] [2] [6 2]
合并[3] [5]
[3 5] [2] [6 2]
合并[3 5] [2]
[2 3 5] [6 2]
第四轮分隔,对[6 2]进行分隔
[2 3 5] [6] [2]
合并[6] [2]
[2 3 5] [2 6]
合并[2 3 5] [2 6]
[2 2 3 5 6]
代码实现(Java)
package com.coder4j.main.arithmetic.sorting;
public class Merge {
private static int mark = 0;
/**
* 归并排序
*
* @param array
* @param low
* @param high
* @return
*/
private static int[] sort(int[] array, int low, int high) {
int mid = (low + high) / 2;
if (low < high) {
mark++;
System.out.println("正在进行第" + mark + "次分隔,得到");
System.out.println("[" + low + "-" + mid + "] [" + (mid + 1) + "-" + high + "]");
// 左边数组
sort(array, low, mid);
// 右边数组
sort(array, mid + 1, high);
// 左右归并
merge(array, low, mid, high);
}
return array;
}
/**
* 对数组进行归并
*
* @param array
* @param low
* @param mid
* @param high
*/
private static void merge(int[] array, int low, int mid, int high) {
System.out.println("合并:[" + low + "-" + mid + "] 和 [" + (mid + 1) + "-" + high + "]");
int[] temp = new int[high - low + 1];
int i = low;// 左指针
int j = mid + 1;// 右指针
int k = 0;
// 把较小的数先移到新数组中
while (i <= mid && j <= high) {
if (array[i] < array[j]) {
temp[k++] = array[i++];
} else {
temp[k++] = array[j++];
}
}
// 两个数组之一可能存在剩余的元素
// 把左边剩余的数移入数组
while (i <= mid) {
temp[k++] = array[i++];
}
// 把右边边剩余的数移入数组
while (j <= high) {
temp[k++] = array[j++];
}
// 把新数组中的数覆盖array数组
for (int m = 0; m < temp.length; m++) {
array[m + low] = temp[m];
}
}
/**
* 归并排序
*
* @param array
* @return
*/
public static int[] sort(int[] array) {
return sort(array, 0, array.length - 1);
}
public static void main(String[] args) {
int[] array = { 3, 5, 2, 6, 2 };
int[] sorted = sort(array);
System.out.println("最终结果");
for (int i : sorted) {
System.out.print(i + " ");
}
}
}
测试输出结果:
正在进行第1次分隔,得到
[0-2] [3-4]
正在进行第2次分隔,得到
[0-1] [2-2]
正在进行第3次分隔,得到
[0-0] [1-1]
合并:[0-0] 和 [1-1]
合并:[0-1] 和 [2-2]
正在进行第4次分隔,得到
[3-3] [4-4]
合并:[3-3] 和 [4-4]
合并:[0-2] 和 [3-4]
最终结果
2 2 3 5 6
经测试,与实例中结果一致。
标签:java归并排序
0
投稿
猜你喜欢
使用Spring Boot进行单元测试详情
2023-11-10 08:01:53
java实现MD5加密的方法小结
2022-02-26 20:01:47
IDEA 单元测试覆盖技巧分享
2022-09-09 03:50:26
redisson 实现分布式锁的源码解析
2022-06-05 05:38:47
java读取文件里面部分汉字内容乱码的解决方案
2022-06-11 03:32:49
Android 图片缩放与旋转的实现详解
2023-03-10 10:57:41
C#如何用ThoughtWorks生成二维码
2022-09-28 20:44:53
@CacheEvict 清除多个key的实现方式
2023-11-21 08:28:04
Struts2学习笔记(9)-Result配置全局结果集
2022-04-09 11:33:10
Java中一些关键字的使用技巧总结
2023-11-19 02:48:49
Android自定义ProgressDialog进度等待框
2022-07-09 13:51:46
C# 实现TXT文档转Table的示例代码
2022-04-23 07:47:52
前端如何调用后端接口进行数据交互详解(axios和SpringBoot)
2023-10-17 02:48:43
深入理解Android热修复技术原理之so库热修复技术
2023-11-19 15:02:10
Spring Boot 2.X快速整合jpa过程解析
2021-09-22 15:27:29
Android实现屏幕录制功能
2022-10-29 15:26:36
Android Spinner 下拉菜单的使用
2023-10-25 10:59:53
Android实现控件拖动效果
2021-11-05 09:02:31
详解Java 中的三种代理模式
2023-06-25 07:49:17
Java文件断点续传实现原理解析
2022-08-21 02:05:39