Java归并排序的递归与非递归实现

时间:2022-01-21 13:57:29

该命题已有无数解释,备份修改后的代码

平均时间复杂度: O(NLogN)  以2为底

最好情况时间复杂度: O(NLogN)

最差情况时间复杂度: O(NLogN)

所需要额外空间: 递归:O(N + LogN), 非递归:O(N)

稳定性: 稳定

归并排序基于分治(快排也是),利用归并来实现排序,其基本思想是:

  如果一个数组有n个数据,则可以把这个数组看作n个有序的子序列,每个子序列的长度为1,然后两两归并,就能得到[n/2]个长度为2(或者1,落单的)的字序列,再不断地两两归并,直到得到一个长度为n的有序数组。

递归:

public static void main(String[] args) {
int[] nums = {6, 5, 3, 8, 1, 7, 2, 9, 4};
segment(nums, 0, nums.length - 1);
System.out.println(Arrays.toString(nums));
} /**
* 递归切分待排
*
* @param nums 待切分数组
* @param left 待切分最后第一个元素的索引
* @param right 待切分数组最后一个元素的索引
*/
private static void segment(int[] nums, int left, int right) {
if (left < right) {
// 找出中间索引
int center = (left + right) / 2;
// 对左边数组进行递归
segment(nums, left, center);
// 对右边数组进行递归
segment(nums, center + 1, right);
// 合并
merge(nums, left, center, right);
}
} /**
* 两两归并排好序的数组(2路归并)
*
* @param nums 带排序数组对象
* @param left 左边数组的第一个索引
* @param center 左数组的最后一个索引,center + 1右数组的第一个索引
* @param right 右数组的最后一个索引
*/
private static void merge(int[] nums, int left, int center, int right) {
int[] tmpArray = new int[right - left + 1];
int leftIndex = left; //左数组第一个元素的索引
int rightIndex = center + 1; //右数组第一个元素索引
int tmpIndex = 0; //临时数组索引 // 把较小的数先移到新数组中
while (leftIndex <= center && rightIndex <= right) {
if (nums[leftIndex] <= nums[rightIndex]) {
tmpArray[tmpIndex++] = nums[leftIndex++];
} else {
tmpArray[tmpIndex++] = nums[rightIndex++];
}
} // 把左边剩余的数移入数组
while (leftIndex <= center) {
tmpArray[tmpIndex++] = nums[leftIndex++];
} // 把右边边剩余的数移入数组
while (rightIndex <= right) {
tmpArray[tmpIndex++] = nums[rightIndex++];
} // 把新数组中的数覆盖nums数组
/*for (int i = 0; i < tmpArray.length; i++) {
nums[begin + i] = tmpArray[i];
}*/
//可以优化成下面的写法
System.arraycopy(tmpArray, 0, nums, left, tmpArray.length);
}

非递归:

public static void main(String[] args) {
int[] nums = {6, 5, 3, 8, 1, 7, 2, 9, 4};
for (int i = 1; i <= nums.length; i *= 2) {
for (int j = 0; j + i <= nums.length; j += i * 2) {
//Math.min 的目的是处理 整个待排序数组为奇数的情况
merge(nums, j, j + i - 1, Math.min(j + 2 * i - 1, nums.length - 1));
}
}
System.out.println(Arrays.toString(nums));
} /**
* 两两归并排好序的数组(2路归并)
*
* @param nums 带排序数组对象
* @param left 左边数组的第一个索引
* @param center 左数组的最后一个索引,center + 1右数组的第一个索引
* @param right 右数组的最后一个索引
*/
private static void merge(int[] nums, int left, int center, int right) {
int[] tmpArray = new int[right - left + 1];
int leftIndex = left; //左数组第一个元素的索引
int rightIndex = center + 1; //右数组第一个元素索引
int tmpIndex = 0; //临时数组索引 // 把较小的数先移到新数组中
while (leftIndex <= center && rightIndex <= right) {
if (nums[leftIndex] <= nums[rightIndex]) {
tmpArray[tmpIndex++] = nums[leftIndex++];
} else {
tmpArray[tmpIndex++] = nums[rightIndex++];
}
} // 把左边剩余的数移入数组
while (leftIndex <= center) {
tmpArray[tmpIndex++] = nums[leftIndex++];
} // 把右边边剩余的数移入数组
while (rightIndex <= right) {
tmpArray[tmpIndex++] = nums[rightIndex++];
} // 把新数组中的数覆盖nums数组
/*for (int i = 0; i < tmpArray.length; i++) {
nums[begin + i] = tmpArray[i];
}*/
//可以优化成下面的写法
System.arraycopy(tmpArray, 0, nums, left, tmpArray.length);
}

参考链接:

[1].http://www.cnblogs.com/yulinfeng/p/7072163.html
[2].https://www.cnblogs.com/yulinfeng/p/7078661.html?utm_source=itdadao&utm_medium=referral
[3].https://blog.csdn.net/lll1204019292/article/details/52346199
[4].https://www.jianshu.com/p/39dd1d9b491d

Java归并排序的递归与非递归实现的更多相关文章

  1. 数据结构二叉树的递归与非递归遍历之java&comma;javascript&comma;php实现可编译(1)java

    前一段时间,学习数据结构的各种算法,概念不难理解,只是被C++的指针给弄的犯糊涂,于是用java,web,javascript,分别去实现数据结构的各种算法. 二叉树的遍历,本分享只是以二叉树中的先序 ...

  2. 二叉树3种递归和非递归遍历(Java)

    import java.util.Stack; //二叉树3种递归和非递归遍历(Java) public class Traverse { /******************一二进制树的定义*** ...

  3. JAVA递归、非递归遍历二叉树(转)

    原文链接: JAVA递归.非递归遍历二叉树 import java.util.Stack; import java.util.HashMap; public class BinTree { priva ...

  4. java扫描文件夹下面的所有文件(递归与非递归实现)

    java中扫描指定文件夹下面的所有文件扫描一个文件夹下面的所有文件,因为文件夹的层数没有限制可能多达几十层几百层,通常会采用两种方式来遍历指定文件夹下面的所有文件.递归方式非递归方式(采用队列或者栈实 ...

  5. AJPFX&colon;递归与非递归之间的转化

    在常规表达式求值中: 输入为四则运算表达式,仅由数字.+.-.*./ .(.) 组成,没有空格,要求求其值. 我们知道有运算等级,从左至右,括号里面的先运算,其次是* ./,再是+.- : 这样我们就 ...

  6. C语言实现 二分查找数组中的Key值&lpar;递归和非递归&rpar;

    基本问题:使用二分查找的方式,对数组内的值进行匹配,如果成功,返回其下标,否则返回 -1.请使用递归和非递归两种方法说明. 非递归代码如下: #include <stdio.h> int ...

  7. 二叉树之AVL树的平衡实现&lpar;递归与非递归&rpar;

    这篇文章用来复习AVL的平衡操作,分别会介绍其旋转操作的递归与非递归实现,但是最终带有插入示例的版本会以递归呈现. 下面这张图绘制了需要旋转操作的8种情况.(我要给做这张图的兄弟一个赞)后面会给出这八 ...

  8. 汉诺塔算法的递归与非递归的C以及C&plus;&plus;源代码

    汉诺塔(又称河内塔)问题其实是印度的一个古老的传说. 开天辟地的神勃拉玛(和中国的盘古差不多的神吧)在一个庙里留下了三根金刚石的棒,第一根上面套着64个圆的金片,最大的一个在底下,其余一个比一 个小, ...

  9. C实现二叉树(模块化集成,遍历的递归与非递归实现)

    C实现二叉树模块化集成 实验源码介绍(源代码的总体介绍):header.h : 头文件链栈,循环队列,二叉树的结构声明和相关函数的声明.LinkStack.c : 链栈的相关操作函数定义.Queue. ...

  10. 简单迷宫算法(递归与非递归C&plus;&plus;实现)

    假定迷宫如下:1代表墙,0代表道路,起点在(1,1),终点(11,9)(PS:下标从0开始计算). 现在寻求一条路径能从起点到达终点(非最短). 有两种解法:递归与非递归. 递归算法思路: 要用递归, ...

随机推荐

  1. JsonView Tool

  2. Linux命令行下创建纳入版本控制下的新目录

    [root@ok 资料库]# svn mkdir test A test [root@ok 资料库]# svn ci -m "mkdir test" Adding 资料库/test ...

  3. DB,Cache和Redis应用场景分析

    最近做一产品,微博方面的.数据存储同时用到了DB(mysql),Cache(memcache),Redis.其实最开始架构设计的时候是准备用MongoDB的,由于学习成本太高,最终选择放弃了,采用了比 ...

  4. 《Genesis-3D开源游戏引擎完整实例教程-跑酷游戏篇01:道路的自动生成》

    1.道路的自动生成 道路自动生成概述: 3D跑酷游戏的核心就是跑,在跑这一过程中增加趣味性使得游戏具有更多的可玩性.道路的自动生成和*拼接,为游戏增设了更多的不可预见性.这种不可预见性使得玩家在游戏 ...

  5. Linux下安装VNC Server

    操作系统centos6.5,在其之上安装vnc server,可利用windows上的vnc client远程登录. 1. 安装 yum install tigervnc-server.x86_64 ...

  6. unity资源

    unity资源集中贴 1.unity经验之谈 http://jingyan.baidu.com/article/19192ad820f17be53e570715.html 2.百度网盘,分享了一点模型 ...

  7. js的call&lpar;&rpar; ,apply&lpar;&rpar; 两种方法的区别和用法,最白话文的解释,让枯燥滚粗!

    百度了一圈calll()函数和apply()函数,感觉还是糊里糊涂 正好我前几天刚又重新翻了一遍 那本 600多页 的圣经书,我习惯时不时的去打下基础,只是为了用来装逼,给人讲解....(我是有多蛋疼 ...

  8. SaltStack 安装介绍 01

    一.入门指南 1.1 SALTSTACK是什么? The backbone of Salt is the remote execution engine, which creates a high-s ...

  9. java--Iterator迭代问题:集合并发访问异常

    用Iterator对数组进行迭代后,如果在迭代过程中对数组进行增加元素操作(这里iterator本身没有提供增加操作方法)时,就会抛出并发访问异常: 异常如下: Exception in thread ...

  10. 文本分类学习(六) AdaBoost和SVM

    直接从特征提取,跳到了BoostSVM,是因为自己一直在写程序,分析垃圾文本,和思考文本分类用于识别垃圾文本的短处.自己学习文本分类就是为了识别垃圾文本. 中间的博客待自己研究透彻后再补上吧. 因为获 ...