两个数组结果相减_学点算法(三)——数组归并排序

今天来学习归并排序算法。

17349d377b446fbd453b26a9a8178446.png

分而治之

归并算法的核心思想是分而治之,就是将大问题转化为小问题,在解决小问题的基础上,再去解决大问题。将这句话套用到排序中,就是将一个大的待排序区间分为小的待排序区间,对小的排序区间进行排序,然后再去解决大区间的排序,而对小区间进行排序的时候可以继续使用该方法,将小的待排序区间分为小小的待排序区间... ...依次往复。最终将排序区间分到只有一个元素,这个时候,因为一个元素已经就是排好序的,无需继续切分了,然后我们再依照此结果去解决大区间的排序。

假设我们现在有[53, 89, 32, 45, 67, 2, 32, 89, 65, 54]这么一个数组,我们要对它进行归并排序(从小到大排序,此文中均为从小到大排序),整体的过程如下图所示:

0a2591f2f9ad17219b06fd8f1cb29240.png

归并排序算法完整过程

整个算法分为两大阶段,分割阶段归并阶段

分割阶段

1. [53, 89, 32, 45, 67, 2, 32, 89, 65, 54]分为[53, 89, 32, 45, 67]和[2, 32, 89, 65, 54]。

2. [53, 89, 32, 45, 67]分为[53, 89]和[32, 45, 67], [2, 32, 89, 65, 54]分为[2, 32]和[89, 65, 54]。

4. ... ...

5. 数组分割完毕,所有小数组依次为[53],[89] ,[32] ,[45],[67],[2],[32],[89],[65],[54]。

归并阶段

1. [53],[89]归并为[53, 89],[32] ,[45]归并为[32, 45],[2]和[32]归并为[2, 32],[65]和[54]归并为[54, 65](这一步中,[67]和[89]没有归并,因为在最后一步分割过程中,它们被单独分开了)。

2. [32, 45]和[67]归并为[32, 45, 67],[89]和[54, 65]归并为[54, 65, 89]。

3. [53, 89]和[32, 45, 67]归并为[32, 45, 53, 67, 89],[2, 32]和[54, 65, 89]归并为[2, 32, 54, 65, 89]。

4. [32, 45, 53, 67, 89]和[2, 32, 54, 65, 89]归并为[2, 32, 32, 45, 53, 54, 65, 67, 89, 89](其中两个32和两个89,在归并的过程中保留它们的原始顺序)。

整个分而治之的过程我们已经清楚了,可还有一个问题没有解决,就是具体应该怎么去归并呢,即如何将两个排序子数组(或子区间)合并为大的排序好的数组(或区间)呢?

我们可以先举个简单的例子:现在有[2]和[1]两个数组,我们如何把它们合并为[1, 2]整个数组呢?很简单,我们首先会把这两个元素取出来,对比一下,取出2和1,我们一对比,发现1小于2, 所以我们在结果数组中先放入1,然后再放入2。可以发现,我们就是将两个子数组中的元素取出来比较了一下,哪个小就把哪个先放入结果数组中。

从上面的例子中我们可以得到大概的思路就是,针对两个有序的子数组(或子区间),我们可以从头依次取两个子数组(或子区间)的首元素(因为从小到大排序后首元素肯定最小),然后作对比,把小的元素放入结果数组中,并且这个元素在下次选取的时候剔除,下一个元素也应用同样的方法得到,放入结果数组中,依次进行,直到两个数组的元素都取完为止,如果发现其中一个子数组(或子区间)率先取完,就直接将另外一个子数组(或子区间)中剩下的元素全部放入结果数组中即可。具体步骤描述如下:

1. 判断两个子数组(或子区间)是否含有剩余元素,如果都有剩余元素,进入第2步;如果只有一个有剩余元素,进入第5步;如果没有,则退出。

2. 取出左子数组(或左子区间)的首个元素和右子数组(或右子区间)的首个元素。

3. 两个元素对比,将小的元素放入结果数组,并且从对应数组中剔除该元素。

4. 回到第1步(上一步选中的元素已被剔除)。

5. 将剩余元素直接全部放入结果数组中,退出(因为元素全部移动完毕)。

其中,剔除这一步在代码实现中可看成索引的移动。

上述这个过程我们取[53, 89]和[32, 45, 67]这两个子数组的合并来描述一下,如图所示:

db6e5363a2ecdcd1e6d3bdf1064b6aa7.png

归并

1. 取出左子数组中的首个元素53和右子数组中的首个元素32,两个作对比,发现32 < 53,所以我们将32放入结果数组:

07125af4005bed9a00fc01e7fef2d940.png

2. 取出左子数组中的首个元素53和右子数组中的首个元素45,两个作对比,发现45 < 53,所以我们将45放入结果数组:

d578d1e3c7860efd12245e39f75958b4.png

3. 取出左子数组中的首个元素53和右子数组中的首个元素67,两个作对比,发现53 < 67,所以我们将53放入结果数组:

4702f5fe45e96658536c6c8aca80e517.png

4. 取出左子数组中的首个元素89和右子数组中的首个元素67,两个作对比,发现67 < 89,所以我们将67放入结果数组:

f159ef56201fd2f7db99b542f82b4163.png

5. 此时我们发现只有左子数组存在元素,所以直接将左子数组的剩下所有元素,此时只有89放入结果数组:

2f81ad7ee06b10875fc6e289158b3106.png

6. 至此,所有元素移动完毕,退出。

通过以上分析,我们可以知道整个归并排序算法总体上分为一个整体的大逻辑(分而治之)和一个局部的小逻辑(归并),在大逻辑(分而治之,将整个数组切分,并在确认子数组有序后归并)的基础上,结合使用小逻辑(归并,将两个有序子数组归并为一个大的有序数组)即可实现对整个数组的排序。

最终代码实现如下:

/** * 数组的归并排序算法 * * @param nums 数组 * @param lo 区间的lo索引(包含) * @param hi 区间的hi索引(不包含) */public static void mergeSort(int[] nums, int lo, int hi) {    // 数组为null则直接返回    if (nums == null) {        return;    }    // 索引检查    if (lo < 0 || nums.length <= lo) {        throw new IllegalArgumentException("lo索引必须大于0并且小于数组长度,数组长度:" + nums.length);    }    if (hi < 0 || nums.length < hi) {        throw new IllegalArgumentException("hi索引必须大于0并且小于等于数组长度,数组长度:" + nums.length);    }    if (hi <= lo) {        // lo索引必须小于hi索引(等于也不行,因为区间是左闭右开,如果等于,区间内元素数量就为0了)        throw new IllegalArgumentException("lo索引必须小于hi索引");    }    if (lo + 1 >= hi) {        // 区间元素个数最多为1        // 无需排序        return;    }    int mid = (lo + hi) / 2;    // 对左子区间排序    mergeSort(nums, lo, mid);    // 对右子区间排序    mergeSort(nums, mid, hi);    // 对两个排序好的子区间归并,得到一个整体有序的区间    merge(nums, lo, mid, hi);}public static void merge(int[] nums, int lo, int mid, int hi) {    // 这里不用检查索引,调用方已经决定了索引是有效的    // 结果区间和右子区间使用原有数组    // 左子区间使用临时数组(因为结果区间可能会覆盖左子区间的元素,所以需要开辟新数组保存)    int leftLen = mid - lo;    int[] left = new int[leftLen];    System.arraycopy(nums, lo, left, 0, leftLen);    // 左子区间索引    int leftIdx = 0;    // 右子区间索引    int rightIdx = mid;    // 结果区间索引    int resultIdx = lo;    while (true) {        if (leftIdx < leftLen && rightIdx < hi) {            // 两个子区间都存在元素            // 取两个子区间的有效首元素对比            if (left[leftIdx] <= nums[rightIdx]) {                // 左子区间首元素小于右子区间首元素                // 将左子区间首元素放到结果位置,同时更新索引位置                nums[resultIdx++] = left[leftIdx++];            } else {                // 右子区间首元素小于左子区间首元素                // 将右子区间首元素放到结果位置,同时更新索引位置                nums[resultIdx++] = nums[rightIdx++];            }        } else {            if (leftIdx < leftLen) {                // 左子区间还有剩余元素                // 直接将左区间所有元素一起移动到结果位置                System.arraycopy(left, leftIdx, nums, resultIdx, leftLen - leftIdx);            } else {                // 右子区间还有剩余元素                // 因为经过上一次判断,左子区间和右子区间只会有一个存在剩余元素                // 直接将右区间所有元素一起移动到结果位置                System.arraycopy(nums, rightIdx, nums, resultIdx, hi - rightIdx);            }            // 全部元素移动完毕,退出            break;        }    }}

测试代码如下:

List numList = IntStream.range(0, 10).boxed().collect(Collectors.toList());for (int i = 1; i <= 5; i++) {    System.out.println("================第" + i + "次================");    Collections.shuffle(numList);    int[] nums = new int[numList.size()];    for (int j = 0; j < nums.length; j++) {        nums[j] = numList.get(j);    }    System.out.println("排序前:" + Arrays.toString(nums));    mergeSort(nums, 0, numList.size());    System.out.println("排序后:" + Arrays.toString(nums));}

运行结果如下:

================第1次================排序前:[8, 4, 1, 6, 7, 0, 5, 9, 2, 3]排序后:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]================第2次================排序前:[2, 5, 6, 7, 9, 4, 3, 1, 0, 8]排序后:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]================第3次================排序前:[2, 0, 5, 6, 7, 3, 4, 9, 8, 1]排序后:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]================第4次================排序前:[4, 0, 3, 8, 1, 5, 9, 7, 2, 6]排序后:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]================第5次================排序前:[7, 9, 8, 2, 0, 5, 6, 3, 4, 1]排序后:[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

测试代码中5次随机将数组打乱,然后运行我们的归并排序算法,均得到有序结果,符合我们的预期。

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

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

相关文章

python实习生面试题_大数据分析实习生面试题库

原标题&#xff1a;大数据分析实习生面试题库大数据分析是一个有吸引力的领域&#xff0c;因为它不仅有利可图&#xff0c;而且您有机会从事有趣的项目&#xff0c;而且您总是在学习新事物。如果您想从头开始&#xff0c;请查看大数据分析实习生面试题库以准备面试要点。大数据…

JavaScript编程语言 基础 (1)

问题&#xff1a;什么是web前端前端&#xff1a;指界面&#xff0c;计算机&#xff08;PC&#xff09;软件桌面的界面&#xff1b; 计算机端的浏览器界面&#xff1b; 移动端的软件&#xff08;app&#xff09;界面&#xff1b; 移动端的浏览器界面。HtmlcssJavaScript 使用网页…

shell结合expect写的批量scp脚本工具

转载链接&#xff1a;http://www.jb51.net/article/34005.htm expect用于自动化地执行linux环境下的命令行交互任务&#xff0c;例如scp、ssh之类需要用户手动输入密码然后确认的任务。有了这个工具&#xff0c;定义在scp过程中可能遇到的情况&#xff0c;然后编写相应的处理语…

ASP记数器

这两天有好几个老的ASP网站要改&#xff0c;其中有要求加记数器&#xff0c;为图简单&#xff0c;就用文本文件的形式存储记数。以前用ifream的形式嵌入&#xff0c;不能很好的控制记数器显示的风格&#xff0c;现在改进了一下&#xff0c;可以很好的与嵌入板块风格结合了。把做…

php利用openssl实现RSA非对称加密签名

转载链接&#xff1a;http://liuxufei.com/weblog/jishu/376.html 1. 先用php生成一对公钥和私钥 $res openssl_pkey_new(); openssl_pkey_export($res,$pri); $d openssl_pkey_get_details($res); $pub $d[key]; var_dump($pri,$pub); 2. 保存好自己的私钥&#xff0c;把公…

[转] DevExpress 第三方控件汉化的全部代码和使用方法

DevExpress.XtraEditors.Controls 此控件包中包含的控件最多&#xff0c;包括文本框&#xff0c;下拉列表&#xff0c;按钮&#xff0c;等等 DevExpress.XtraGrid 网格 DevExpress.XtraBars 菜单栏 和 工具栏 DevExpress.XtraNavBar 导航条 DevExpress.XtraPr…

QPM 性能监控组件总篇

QPM &#xff08;Quality Performance Monitor&#xff09; 是一个质量性能监控组件&#xff0c;可以很方便的查看当前 App 的性能和常用数据。目前主要运行在 Android 平台上&#xff0c;通过集成 QPM 组件&#xff0c;可以在 App 中通过悬浮窗可视化相关实时数据。意在帮助广…

福音!微信个人公众号可以改名了!

微信个人公众号可以改名了&#xff01;&#xff01;&#xff01;今年&#xff0c;我们学校从景德镇陶瓷学院更名为景德镇陶瓷大学&#xff0c;但苦于微信限制&#xff0c;很多微信公众号无法更名。很多组织社团就放弃了原先的关注量&#xff0c;重新申请注册账号。当前我们的订…

js list删除指定元素_删除js数组中的指定元素,有这两步就够了

js数组是js部分非常重要的知识&#xff0c;有时我们有这么个需求js数组删除指定元素&#xff0c;先定义一个函数来获取删除指定元素索引值&#xff0c;然后用js数组删除的方法&#xff0c;来删除指定元素即可&#xff0c;就两步不难&#xff0c;很简单。1、JS的数组对象定义一个…

sudo 安装 常见错误

运行环境Linux&#xff1a; 1、sudo&#xff1a;安装 apt-get install sudo 2、sudo: must be setuid root错误解决方法. ls -l /usr/bin/sudo chown root:root /usr/bin/sudo chmod 4755 /usr/bin/sudo reboot 3、sudo&#xff1a;提示用户无权限之类 在 /etc/…

慕课网高并发实战(一)-并发与高并发基本概念

课程网址 并发&#xff1a; 同时拥有两个或者多个线程&#xff0c;如果程序在单核处理器上运行&#xff0c;多个线程交替得换入或者换出内存&#xff0c;这些线程是同时“存在”的&#xff0c;每个线程都处于执行过程中的某个状态&#xff0c;如果运行在多核处理器上&#xff…

2009最经典名句

一&#xff1a;我的优点是&#xff1a;我很帅&#xff1b;但是我的缺点是&#xff1a;我帅的不明显. 二&#xff1a;谈钱不伤感情&#xff0c;谈感情最他妈伤钱。 三&#xff1a;我诅咒你一辈子买方便面没有调料包。 四&#xff1a;会计说&#xff1a;“你晚点来领工资吧&#…

计算机协会丨让技能得到提升,让思维受到启迪

“ 各位2016级新生&#xff0c;新的学期马上就要开始了&#xff0c;学校的各个组织和社团你真的了解了吗&#xff1f;在眼花缭乱的社团里如何找到自己真正喜欢的呢&#xff1f;或许看完计算机协会的纳新微信你就都明白啦&#xff01;关键词&#xff1a;计算机协会景德镇陶瓷大学…

ondestroy什么时候调用_尾调用和尾递归

尾调用1. 定义尾调用是函数式编程中一个很重要的概念&#xff0c;当一个函数执行时的最后一个步骤是返回另一个函数的调用&#xff0c;这就叫做尾调用。注意这里函数的调用方式是无所谓的&#xff0c;以下方式均可&#xff1a;函数调用: func()方法调用: obj.method()call调用:…

查看/修改Linux时区和时间

转载链接&#xff1a;http://blog.csdn.net/colincjl/article/details/6133036 查看/修改Linux时区和时间 一、时区 1. 查看当前时区 date -R 2. 修改设置时区 方法(1) tzselect 方法(2) 仅限于RedHat Linux 和 CentOS timeconfig 方法(3) 适用于Debian dpkg-reconfigure tzdat…

dhl:使用return RedirectToAction()和 return view()

一个Action&#xff1a; Code/// <summary> /// Friend好友的地 /// </summary> /// <returns></returns> public ActionResult FriendFarm(string pid) {BLL.DTOFarm farm new AppleGrange.BLL.DTOFarm(pid); …

【更名通知】将以个人名义继续更新维护

这是我&#xff08;2013年任职计算机协会会长&#xff09;在2013年申请的公众号。由于2016年学校陶院更名为陶大&#xff0c;在当时公众号无法修改名称。后来计协的的学弟学妹申请了新的公众号"陶大计算机Association"&#xff0c;大家可以前往关注&#xff0c;所以该…

CentOS7.6 MySQL8环境搭建 配置远程登录 字符集UTF8 简单密码

一、环境准备 1、清理环境中系统自带的MySQL &#xff08;1&#xff09;删除系统自带的MySQL或Mariadb yum remove mysql-libs &#xff08;2&#xff09;查询系统中是否还有残余的依赖包 rpm -qa | grep mariadb &#xff08;3&#xff09;删除rpm依赖包 rpm -e --nodeps mar…

radio切换控制div显示_JavaScript连载31图片动态切换以及关闭图片案例

一、图标切换31.1点击那两个按钮可以做到轮番显示图片二、关闭图片案例31.2点击右上角的叉&#xff0c;图片会消失。三、源码&#xff1a;D31_iconSwitch.htmlD31_2_CloseImage.html地址:https://github.com/ruigege66/JavaScript/blob/master/D31_iconSwitch.htmlhttps://gith…

jQuery 1.9+ 移除$.browser方法

转载链接&#xff1a;http://blog.csdn.net/czplplp_900725/article/details/8704438 jQuery 从 1.9 版开始&#xff0c;移除了 $.browser 和 $.browser.version &#xff0c;取而代之的是 $.support。 在更新的 2.0 版本中&#xff0c;将不再支持 IE 6/7/8。 以后&#xff0c;…