Java排序之归并排序
Java排序之归并排序
1. 简介
归并排序的算法是将多个有序数据表合并成一个有序数据表。如果参与合并的只有两个有序表,则成为二路合并。对于一个原始的待排序数列,往往可以通过分割的方法来归结为多路合并排序。
2. 归并排序思路
- 将长度为n的待排序数组看做是由n个有序长度为1的数组组成
- 将其两两合并,得到长度为2的有序数组
- 然后再对这些子表进行合并,得到长度为4的有序数组
- 重复上述过程,一直到最后的子表长度为n也就完成了排序
3. 代码实例
归并排序有两种实现方式:递归和非递归。在看归并排序的代码之前先来看一下怎么和合并两个有序数组:
// 基础,合并两个有序数组
public static int[] merge2Arr(int[] arr1, int[] arr2) {
int len1 = arr1.length;
int len2 = arr2.length;
int[] res = new int[len1 + len2]; // 使用一个数组用来存储排好序的数组
int i = 0, j = 0, k = 0;
while(i < len1 && j < len2) {
res[k++] = arr1[i] < arr2[j]? arr1[i++] : arr2[j++];
}
while(i < len1) {
res[k++] = arr1[i++];
}
while(j < len2) {
res[k++] = arr2[j++];
}
return res;
}
归并排序的递归实现:
// 归并排序,递归实现
public void sortMergeRecursion(int[] nums) {
sortMergeRecursionHelper(nums, 0, nums.length - 1);
}
public void sortMergeRecursionHelper(int[] nums,int left, int right) {
if(left == right) return; // 当待排序的序列长度为1时,递归开始回溯,进行merge
int middle = left + (right - left) / 2;
sortMergeRecursionHelper(nums, left, middle);
sortMergeRecursionHelper(nums, middle + 1, right);
mergeArr(nums, left, middle, right);
}
public void mergeArr(int[] nums, int left, int middle, int right) {
int[] tem = new int[right - left + 1];
int i = left, j = middle + 1, k = 0;
while(i <= middle && j <= right) {
tem[k++] = nums[i] < nums[j]? nums[i++] : nums[j++];
}
while(i <= middle) {
tem[k++] = nums[i++];
}
while(j <= right) {
tem[k++] = nums[j++];
}
// 将辅助数组数据写入原数组
int index = 0;
while(left <= right) {
nums[left++] = tem[index++];
}
}
归并排序的非递归实现(迭代):
// 归并排序,非递归实现(迭代)
public void sortMergeIteration(int[] nums) {
int len = 1; // 初始排序数组的长度
while(len < nums.length) {
for(int i = 0; i < nums.length; i += len * 2) {
sortMergeIterationHelper(nums, i, len);
}
len *= 2; // 每次将排序数组的长度*2
}
}
/**
* 辅助函数
* @param nums 原数组
* @param start 从start位置开始
* @param len 本次合并的数组长度
*/
public void sortMergeIterationHelper(int[] nums, int start, int len) {
int[] tem = new int[len * 2];
int i = start;
int j = start + len;
int k = 0;
while(i < start + len && (j < start + len + len && j < nums.length)) {
tem[k++] = nums[i] < nums[j]? nums[i++] : nums[j++];
}
while(i < start + len && i < nums.length) { // 注意:这里i也可能超出长度
tem[k++] = nums[i++];
}
while(j < start + len + len && j < nums.length) {
tem[k++] = nums[j++];
}
int right = start + len + len;
int index = 0;
while(start < nums.length && start < right) {
nums[start++] = tem[index++];
}
}
归并排序的时间复杂度为O(n*log2n),空间复杂度为O(n)
归并排序是一种稳定的排序方法。
参考:
Java排序之归并排序的更多相关文章
- 【java排序】 归并排序算法、堆排序算法
一.归并排序算法 基本思想: 归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的.然后再把有序子序列合并为整体有序序列. 归并 ...
- Java排序算法——归并排序
import java.util.Arrays; //================================================= // File Name : MergeSor ...
- java排序算法-归并排序
public class MergeSort { private static void mergeSortTest() { int[] in = { 2, 5, 3, 8, 6, 7, 1, 4, ...
- java排序算法(九):归并排序
java排序算法(九):归并排序
- java泛型中使用的排序算法——归并排序及分析
一.引言 我们知道,java中泛型排序使用归并排序或TimSort.归并排序以O(NlogN)最坏时间运行,下面我们分析归并排序过程及分析证明时间复杂度:也会简述为什么java选择归并排序作为泛型的排 ...
- 程序员必知的8大排序(四)-------归并排序,基数排序(java实现)
程序员必知的8大排序(一)-------直接插入排序,希尔排序(java实现) 程序员必知的8大排序(二)-------简单选择排序,堆排序(java实现) 程序员必知的8大排序(三)-------冒 ...
- java排序集锦
java实现排序的一些方法,来自:http://www.javaeye.com/topic/548520 package sort; import java.util.Random; /** * 排序 ...
- (转)JAVA排序汇总
JAVA排序汇总 package com.softeem.jbs.lesson4; import java.util.Random; /** * 排序测试类 * * 排序算法的分类如下: * 1.插入 ...
- java排序算法(一):概述
java排序算法(一)概述 排序是程序开发中一种非常常见的操作,对一组任意的数据元素(活记录)经过排序操作后,就可以把它们变成一组按关键字排序的一组有序序列 对一个排序的算法来说,一般从下面三个方面来 ...
随机推荐
- 题解 P3378 【【模板】堆】
Update 18.2.27----想当年我还用着C..... 看到题解里一堆用C++ STL库中的优先队列,身为C语言选手心里不是滋味 故手打一个优先队列献给坚守在C语言的选手 #include & ...
- SQL递归查询(with as)
SQL递归查询(with cte as) with cte as( select Id,Pid,DeptName,0 as lvl from Department where Id = 2 ...
- vue-filters(过滤器)
局部过滤器: <html> <head> <title>vue</title> <meta charset="utf-8"&g ...
- keras 切换后端 TensorFlow,cntk,theano
参考 https://keras.io/#configuring-your-keras-backend https://keras.io/backend/ Switching from one bac ...
- 部署在SAP Cloud Platform CloudFoundry环境的应用如何消费SAP Leonardo机器学习API
Jerry的前一篇文章 如何在Web应用里消费SAP Leonardo的机器学习API 里介绍的例子是Neo测试环境的Web应用消费sandbox版本的机器学习API,url如下: https://s ...
- Java之IO学习
知其然也要知其所以然,最近对IO流莫名的感觉时候充充电了,所以上网拜读了大神的文章.在学习过程中我找到了一篇很系统很详细的文章贴出来. 声明一下这是转载的文章 原文:https://www.cnbl ...
- win10开机后将存在多个系统选择,改为直接进入系统无需选择
win10系统安装后,可能出现每次开机都要选择操作系统,比较麻烦,所以就来设置下如何直接进入系统,无须选择 1.我的电脑右键“属性”—“高级系统设置”—“系统属性” 2.设置“启动和故障恢复”如下 选 ...
- 索引 _id
_id索引是绝大多数集合默认建立的索引,对于每个插入的数据,mongodb都会自动生成一条唯一的_id字段 增加一个数据 > db.test2.insert({x:1}) WriteResult ...
- Python3+Appium学习笔记07-元素定位工具UI Automator Viewer
这篇主要说下如何使用UI Automator Viewer这个工具来定位元素.这个工具是sdk自带的.在sdk安装目录Tools目录下找到uiautomatorviewer.bat并启动它 如果启 ...
- php高精度计算