一、归并排序

归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

归并过程为:比较a[i]和a[j]的大小,若a[i]≤a[j],则将第一个有序表中的元素a[i]复制到r[k]中,并令i和k分别加上1;否则将第二个有序表中的元素a[j]复制到r[k]中,并令j和k分别加上1,如此循环下去,直到其中一个有序表取完,然后再将另一个有序表中剩余的元素复制到r中从下标k到下标t的单元。归并排序的算法我们通常用递归实现,先把待排序区间[s,t]以中点二分,接着把左边子区间排序,再把右边子区间排序,最后把左区间和右区间用一次归并操作合并成有序的区间[s,t]。

二、归并操作

三、两路归并算法

1、算法基本思路

设两个有序的子文件(相当于输入堆)放在同一向量中相邻的位置上:R[low..m],R[m+1..high],先将它们合并到一个局部的暂存向量R1(相当于输出堆)中,待合并完成后将R1复制回R[low..high]中。

(1)合并过程

合并过程中,设置i,j和p三个指针,其初值分别指向这三个记录区的起始位置。合并时依次比较R[i]和R[j]的关键字,取关键字较小的记录复制到R1[p]中,然后将被复制记录的指针i或j加1,以及指向复制位置的指针p加1。

重复这一过程直至两个输入的子文件有一个已全部复制完毕(不妨称其为空),此时将另一非空的子文件中剩余记录依次复制到R1中即可。

(2)动态申请R1

实现时,R1是动态申请的,因为申请的空间可能很大,故须加入申请空间是否成功的处理。

2、归并算法

void Merge(SeqList R,int low,int m,int high)

{//将两个有序的子文件R[low..m)和R[m+1..high]归并成一个有序的

//子文件R[low..high]

int i=low,j=m+1,p=0; //置初始值

RecType *R1; //R1是局部向量,若p定义为此类型指针速度更快

R1=(ReeType *)malloc((high-low+1)*sizeof(RecType));

if(! R1) //申请空间失败

Error("Insufficient memory available!");

while(i<=m&&j<=high) //两子文件非空时取其小者输出到R1[p]上

R1[p++]=(R[i].key<=R[j].key)?R[i++]:R[j++];

while(i<=m) //若第1个子文件非空,则复制剩余记录到R1中

R1[p++]=R[i++];

while(j<=high) //若第2个子文件非空,则复制剩余记录到R1中

R1[p++]=R[j++];

for(p=0,i=low;i<=high;p++,i++)

R[i]=R1[p];//归并完成后将结果复制回R[low..high]

} //Merge

四、归并排序

归并排序有两种实现方法:自底向上和自顶向下。下面说说自顶向下的方法

(1)分治法的三个步骤

设归并排序的当前区间是R[low..high],分治法的三个步骤是:

①分解:将当前区间一分为二,即求分裂点

②求解:递归地对两个子区间R[low..mid]和R[mid+1..high]进行归并排序;

③组合:将已排序的两个子区间R[low..mid]和R[mid+1..high]归并为一个有序的区间R[low..high]。

递归的终结条件:子区间长度为1(一个记录自然有序)。

(2)具体算法

void MergeSortDC(SeqList R,int low,int high)

{//用分治法对R[low..high]进行二路归并排序

int mid;

if(low

mid=(low+high)/2; //分解

MergeSortDC(R,low,mid); //递归地对R[low..mid]排序

MergeSortDC(R,mid+1,high); //递归地对R[mid+1..high]排序

Merge(R,low,mid,high); //组合,将两个有序区归并为一个有序区

}

}//MergeSortDC

(3)算法MergeSortDC的执行过程

算法MergeSortDC的执行过程如下图所示的递归树。

五、算法分析

1、稳定性

归并排序是一种稳定的排序。

2、存储结构要求

可用顺序存储结构。也易于在链表上实现。

3、时间复杂度

对长度为n的文件,需进行 趟二路归并,每趟归并的时间为O(n),故其时间复杂度无论是在最好情况下还是在最坏情况下均是O(nlgn)。

4、空间复杂度

需要一个辅助向量来暂存两有序子文件归并的结果,故其辅助空间复杂度为O(n),显然它不是就地排序。

注意:

若用单链表做存储结构,很容易给出就地的归并排序。

5、比较操作的次数介于(nlogn) / 2和nlogn - n + 1。

6、赋值操作的次数是(2nlogn)。归并算法的空间复杂度为:0 (n)

7、归并排序比较占用内存,但却是一种效率高且稳定的算法。

六、代码实现

public class MergeSortTest {

public static void main(String[] args) {

int[] data = new int[] { 2, 4, 7, 5, 8, 1, 3, 6 };

System.out.print("初始化:\t");

print(data);

System.out.println("");

mergeSort(data, 0, data.length - 1);

System.out.print("\n排序后: \t");

print(data);

}

public static void mergeSort(int[] data, int left, int right) {

if (left >= right)

return;

//两路归并

// 找出中间索引

int center = (left + right) / 2;

// 对左边数组进行递归

mergeSort(data, left, center);

// 对右边数组进行递归

mergeSort(data, center + 1, right);

// 合并

merge(data, left, center, center + 1, right);

System.out.print("排序中:\t");

print(data);

}

/**

* 将两个数组进行归并,归并前面2个数组已有序,归并后依然有序

*

* @param data

* 数组对象

* @param leftStart

* 左数组的第一个元素的索引

* @param leftEnd

* 左数组的最后一个元素的索引

* @param rightStart

* 右数组第一个元素的索引

* @param rightEnd

* 右数组最后一个元素的索引

*/

public static void merge(int[] data, int leftStart, int leftEnd,

int rightStart, int rightEnd) {

int i = leftStart;

int j = rightStart;

int k = 0;

// 临时数组

int[] temp = new int[rightEnd - leftStart + 1]; //创建一个临时的数组来存放临时排序的数组

// 确认分割后的两段数组是否都取到了最后一个元素

while (i <= leftEnd && j <= rightEnd) {

// 从两个数组中取出最小的放入临时数组

if (data[i] > data[j]) {

temp[k++] = data[j++];

} else {

temp[k++] = data[i++];

}

}

// 剩余部分依次放入临时数组(实际上两个while只会执行其中一个)

while (i <= leftEnd) {

temp[k++] = data[i++];

}

while (j <= rightEnd) {

temp[k++] = data[j++];

}

k = leftStart;

// 将临时数组中的内容拷贝回原数组中 // (原left-right范围的内容被复制回原数组)

for (int element : temp) {

data[k++] = element;

}

}

public static void print(int[] data) {

for (int i = 0; i < data.length; i++) {

System.out.print(data[i] + "\t");

}

System.out.println();

}

}

七、运行结果

初始化: 2 4 7 5 8 1 3 6

排序中: 2 4 7 5 8 1 3 6

排序中: 2 4 5 7 8 1 3 6

排序中: 2 4 5 7 8 1 3 6

排序中: 2 4 5 7 1 8 3 6

排序中: 2 4 5 7 1 8 3 6

排序中: 2 4 5 7 1 3 6 8

排序中: 1 2 3 4 5 6 7 8

排序后: 1 2 3 4 5 6 7 8

归并排序java_Java经典排序算法之归并排序详解相关推荐

  1. python选择排序从大到小_经典排序算法和Python详解之(一)选择排序和二元选择排序...

    本文源自微信公众号[Python编程和深度学习]原文链接:经典排序算法和Python详解之(一)选择排序和二元选择排序,欢迎扫码关注鸭! 扫它!扫它!扫它 排序算法是<数据结构与算法>中最 ...

  2. 归并排序 java 迭代_经典排序算法之归并排序(示例代码)

    归并排序(英语:Merge sort,或mergesort),是创建在归并操作上的一种有效的排序算法,效率为 (大O符号).1945年由约翰·冯·诺伊曼首次提出.该算法是采用分治法(Divide an ...

  3. 万字总结八大排序算法(图文详解)

    目录 一.冒泡排序 基本思想: 主要思路: 二.选择排序 基本思想: 主要思路: 三.插入排序 基本思想: 主要思路: 插入排序优化: 四.希尔排序 基本思想: 思路步骤: 代码示例: 对于希尔排序稳 ...

  4. 选择排序算法与示例详解(c语言)

    选择排序是排序算法的一种,思想就是,每一轮寻找数组中最大的值或者最小的值,放在头部或者放入一个新的数组.这样经历一轮遍历,数组或者新数组就是排好序的,他的目的很明确,每次找最大值或者最小值. 这个放在 ...

  5. JavaScript实现十大排序算法(图文详解)

    冒泡排序 排序的效果图 解法 当前解法为升序 冒泡排序的特点,是一个个数进行处理.第i个数,需要与后续的len-i-1个数进行逐个比较. 为什么是 `len-i-1`个数? 因为数组末尾的i个数,已经 ...

  6. 排序算法之快速排序详解

    一.算法介绍 快速排序:快速排序的基本思想是通过一次排序将等待的记录分成两个独立的部分,其中一部分记录的关键字小于另一部分的关键字.C部分的快速排序一直持续到整个序列被排序. 任取一个元素 (如第一个 ...

  7. python版 常用排序算法 思路加详解 冒泡排序、快速排序、插入排序、选择排序

    注:这里所有排序操作都以从小到大排列为例,想要从大到小排的自行修改代码即可 目录 一.冒泡排序 思路: 步骤: 解析: 二.快速排序 思路: 步骤: 代码: 三.插入排序 思路: 代码: 四.选择排序 ...

  8. NOI提高级:排序算法之归并排序、快速排序

    图解排序算法(四)之归并排序 图解排序算法(四)之归并排序 - dreamcatcher-cx - 博客园 小学生图解排序算法:⑥归并排序 小学生图解排序算法:⑥归并排序_纯文笔记-CSDN博客_图解 ...

  9. 十大经典排序算法-归并排序算法详解

    十大经典排序算法 十大经典排序算法-冒泡排序算法详解 十大经典排序算法-选择排序算法详解 十大经典排序算法-插入排序算法详解 十大经典排序算法-希尔排序算法详解 十大经典排序算法-快速排序算法详解 十 ...

最新文章

  1. Linux中锚定符号的作用,Linux基础(9)文本处理三剑客之grep
  2. Activity的跳转及返回值,activity的生命周期
  3. opencsv : 解析CSV
  4. Hive _分桶及抽样查询
  5. [独家]网易遭遇****** 留下“装B”两字
  6. mybatis源码学习
  7. 日志长度_Kafka 日志存储详解
  8. 翼支付和银行网络连通准备
  9. 从零开始学Pytorch(六)之梯度消失、梯度爆炸
  10. Ansible(二)——基本环境的部署以及常用模块的使用
  11. python 输入一个列表s和一个由二元组成的列表p_re --- 正则表达式操作 — Python 3.9.1 文档...
  12. 计数译码显示电路实验报告总结_译码器及其应用
  13. php 微信支付h5 referer,微信H5支付
  14. 青藏高原的气候类型及气候特征
  15. (二)智能化技术如何赋能能源数字化转型及智慧化应用?
  16. 最新 | 诺奖得主涉嫌论文造假
  17. 网易实探瑞幸门店:消费者1.8折买到手软,店员累到流汗
  18. 再见python你好julia_再见 Python,你好 Julia!
  19. memory_max_target/memory_target设置过大报ORA-00845错误
  20. 无法在web服务器上启动调试。未能启动asp.net调试。在不调试的情况下启动项目也许能获得更多信息。

热门文章

  1. python高级应用_Python高级应用程序设计任务
  2. 区块链关键技术1(笔记)
  3. c语言解引用运算符,C++ 解引用(*)和箭头(-)运算符的重载
  4. [2021.10.14][Android P]OpenCamera详细分析(Camera2+Hal3)
  5. 看泽塔云如何布局自己的超融合之路
  6. JS案例学习——随机点名案例
  7. c++析构函数的调用
  8. 数据字典(Data Dictionary)
  9. Altium Designer基础使用
  10. Java学习之常用的Java构建工具