0

0

Java数据结构之插入排序与希尔排序怎么实现

WBOY

WBOY

发布时间:2023-05-13 15:19:06

|

1580人浏览过

|

来源于亿速云

转载

     一、正文

    1.排序的概念及其运用

    1.1排序的概念
    排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。稳定性:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次 序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排 序算法是稳定的;否则称为不稳定的。内部排序:数据元素全部放在内存中的排序。外部排序:数据元素太多不能同时放在内存中,根据排序过程的要求不能在内外存之间移动数据的排序。
    1.2排序运用

            看过排序的基础概念,可能有的小伙伴会问就算我学会了排序,但是在实际生活中有什么用吗?其实排序在生活中无处不在,比如说对一件商品不同维度的选择,又或者是对高校的排名,其实背后都存在着排序的思想,学好排序,能够帮助我们以另一种维度来观察生活中的方方面面并帮助我们更好地解决生活中的问题。  

    Java数据结构之插入排序与希尔排序怎么实现

    Java数据结构之插入排序与希尔排序怎么实现

    1.3常见的排序算法

    在数据结构这一块,我们常见的排序算法共有四种:

    插入排序:直接插入排序、希尔排序

    立即学习Java免费学习笔记(深入)”;

    选择排序:选择排序、堆排序

    交换排序:冒泡排序、快速排序

    归并排序:归并排序

    2.插入排序算法的实现

            由于篇幅的关系,本篇我们主要介绍的是插入排序中的直接插入排序希尔排序,而直接插入排序又常常被称为插入排序。 

    2.1插入排序

    2.1.1基本思想

            直接插入排序是一种简单的插入排序法

            其基本思想是把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列

            实际中我们玩扑克牌时,就用了插入排序的思想。当你摸了一张新牌,自然而然地就会与手上已有的牌堆进行一一比较,在比较之后将其放入其应该所处的位置。所以我们可能并不知道插入排序是什么,但我们潜意识的做法恰恰就符合了插入排序。

     2.1.2直接插入排序

            用比较书面的语言来描述直接插入排序:当插入第i(i>=1)个元素时,前面的 array[0],array[1],…,array[i-1]已经排好序,此时用array[i]的排序码与 array[i-1],array[i-2],…的排序码顺序进行比较,找到插入位置即将array[i]插入,原来位置上的元素顺序后移

            但这么说可能有的小伙伴会不太理解,那么通俗地来讲吧。现在在你面前有一个乱序的数组,我们的目的是要将这个乱序的数组调整为升序或者降序

            以升序为例叭,由于数组是无序的,因此我们需要从数组的第二个元素开始排序。为什么不是第一个呢,因为只有一个数字的的时候,你无法与其余元素比较,自然也就没有乱序一说,因此只有一个元素的时候我们就默认它是有序的。

            在理解完为什么要从第二个元素开始排序后,现在我们就要进行元素的依次插入和排序了。先是第二个元素的插入和排序,在下图中我们会发现第二个元素是44,44大于第一个元素3,因此不需要挪动第二个元素。紧接着就是第三个元素的插入和排序,我们发现第三个元素38小于第二个元素44,不符合我们升序的预期,因此将44挪动到38的位置,在第二、三个元素有序后,我们发现38大于3,也就是第一、二个元素也是有序的,因此就无需再挪动第一个元素的位置,这时候我们已经确认38应该所处的是数组中第二个元素的位置,因此我们只需将38插入到第二个元素的位置即可。相信看到这里的小伙伴对后续元素的插入与排序应该是信手拈来了,

            接下来就是代码的书写了。在代码上,我们该如何实现上述元素的插入与排序呢?我们采取了两个主要的变量“des”“end”,des就是我们所要插入的元素的初识下标,而end代表的是插入之前的序列的最后一个元素的下标,随着des的比较,end要不断向前移动,那么什么时候end的移动才会停止呢,也就是比较的结束,大致分为两种情况:①des所代表的元素大于end所代表的的元素 ②end已经来到序列的第一个元素,这时候des要么是第一个元素,或者是第二个元素。

            具体图片和代码如下:

    Java数据结构之插入排序与希尔排序怎么实现

    Java数据结构之插入排序与希尔排序怎么实现

    //插入排序[升序]
    int* InsertSort(int* arr, int n)
    {
     
    	//整体排序
    	for (int i = 1; i < n; i++)
    	{
    		int end = i - 1;
    		int des = arr[i];
    		//单趟排序
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + 1] = arr[end];
    				end--;
    			}
    			arr[end+1] = des;
    		}
    	}
    }

     注:直接插入排序特性总结

    AI Content Detector
    AI Content Detector

    Writer推出的AI内容检测工具

    下载

    ①元素集合越接近有序,直接插入排序算法的时间效率越高

    ②时间复杂度:O(N^2)

    ③ 空间复杂度:O(1),它是一种稳定的排序算法

    ④ 稳定性:稳定

    2.1.3希尔排序(缩小增量排序)

            希尔排序法又称缩小增量法。

            希尔排序法的基本思想是:先选定一个整数,把待排序文件中所有记录分成整数个组,所有距离为的记录分在同一组内,并对每一组内的记录进行排序。然后重复上述分组和排序的工作,当到达整数等于1时,所有记录在统一组内排好序。

            通俗来讲,希尔排序就是多次的直接插入排序,不过除了最后一次直接插入排序之外的排序又和原本的直接插入排序有所不同。那么有的小伙伴看到这里可能就会问了为什么要进行多次的插入排序,单次的插入排序又和正常的插入排序不同在哪里呢?别着急,下面我们一个个回答。

            先是为什么要多次的插入排序,看过上面对于插入排序的特性总结我们会发现,当元素的集合越接近有序,那么对其进行插入排序的时间效率就越高。因此希尔排序除了最后一次的排序是正常的插入排序之外的多次插入排序的目的就是不断的调整这个元素的集合,使其不断的接近有序

            紧接着就是希尔排序除最后一次插入排序之外的插入排序与正常插入排序的差异。通过上面对插入排序的学习,我们会发现对于一个乱序的数组的来说,一个元素若想来到正确的位置必须要与其余元素一一比较,也就是一步步的挪动,这种挪动在数组元素个数少的情况下尚可,但当这个数组的元素个数很多的时候,以升序来说,想象这个数组内最大的元素位于数组的第一个位置,那么是不是就要将这个元素与数组内其余元素一一比较以后,才能来到数组的最后一个位置,但当我们加大比较的步伐,也就是增大相比较的两个元素之间的距离,那么这个元素是不是就可以更快的来到它应该所处的位置。放置于飞行棋的情境之下,插入排序每次都掷出1,而哈希排序除了最后一次的插入排序掷出的点数是1,其余的插入排序所掷出的点数是都是大于1的,所以可想而知,哈希排序能够更快的到达排序的终点。

            为了方便小伙伴们的理解,这部分代码共分为两部分:①固定步伐的直接插入排序②哈希排序。

            先是固定步伐的直接插入排序,先让我们通过图片来直观的看到数组数组内的元素通过这种操作后的变化。 

    Java数据结构之插入排序与希尔排序怎么实现

    //固定步伐的直接插入排序[升序]
    void ShellSort(int* arr, int n)
    {
    	int gap = 3;
    	int end;
    	//有两种写法,看你要控制end,还是des
    	/*for (int i=0; i < n-gap; i++)
    	{
    		int end = i;
    		int des = arr[end + gap];
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + gap] = arr[end];
    				end -= gap;
    			}
    			arr[end + gap] = des;
    		}
    	}*/
     
    	for (int i = gap; i < n ; i++)
    	{
    		int end = i-gap;
    		int des = arr[end + gap];
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + gap] = arr[end];
    				end -= gap;
    			}
    			arr[end + gap] = des;
    		}
    	}
    }

             接着就是希尔排序

             上述的代码是gap=3的情况下的直接插入排序,那么对于希尔排序而言,我们该对gap该如何选择呢?对于不同gap值的插入排序来说,我们会发现:gap越大,元素跳得越快,数组越不接近有序;而gap越小,元素跳的越慢,数组越接近有序。由于数组的大小不定,因此希尔排序也没有一个合适gap值适用于所有数组,显然,这个gap值一定是动态变化的。

            对于gap的动态变化,常见的有两种:

    ①令gap等于数组的元素个数,每次插入排序后令gap除等2

    ②另一种则是令gap等于数组的元素个数,不过每次插入排序后令gap除以3再加1

            无论哪种处理都能使gap动态变化并最后等于1,对数组进行一次插入排序,达到最后想要的效果。

            代码如下:

    //希尔排序
    void ShellSortPlus(int* arr, int n)
    {
    	int gap=n;
    	int end;
    	while (gap > 1)
    	{
    	gap = gap / 2;
    		
    		for (int i=0; i < n - gap;i++)//有两种写法,看你要控制end,还是des
    		{
    			int end = i;
    			int des = arr[end + gap];
    			while (end >= 0)
    			{
    				if (des >= arr[end])
    					break;
    				if (des < arr[end])
    				{
    					arr[end + gap] = arr[end];
    					end -= gap;
    				}
    				arr[end + gap] = des;
    			}
    		}
     
    	}
    }

    二、测试代码

            为了方便小伙伴们测试排序后的效果,为大家提供了测试的代码并包含排序的具体代码和包含的头文件。

    #include 
    #include 
    #include 
     
    //插入排序[升序]
    int* InsertSort(int* arr, int n)
    {
     
    	//整体排序
    	for (int i = 1; i < n; i++)
    	{
    		int end = i - 1;
    		int des = arr[i];
    		//单趟排序
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + 1] = arr[end];
    				end--;
    			}
    			arr[end+1] = des;
    		}
    	}
    }
     
    //固定步伐的直接插入排序[升序]
    void ShellSort(int* arr, int n)
    {
    	int gap = 3;
    	int end;
    	//有两种写法,看你要控制end,还是des
    	/*for (int i=0; i < n-gap; i++)
    	{
    		int end = i;
    		int des = arr[end + gap];
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + gap] = arr[end];
    				end -= gap;
    			}
    			arr[end + gap] = des;
    		}
    	}*/
     
    	for (int i = gap; i < n ; i++)
    	{
    		int end = i-gap;
    		int des = arr[end + gap];
    		while (end >= 0)
    		{
    			if (des >= arr[end])
    				break;
    			if (des < arr[end])
    			{
    				arr[end + gap] = arr[end];
    				end -= gap;
    			}
    			arr[end + gap] = des;
    		}
    	}
    }
     
     
    //希尔排序
    void ShellSortPlus(int* arr, int n)
    {
    	int gap=n;
    	int end;
    	while (gap > 1)
    	{
    	gap = gap / 2;
    		
    		for (int i=0; i < n - gap;i++)//有两种写法,看你要控制end,还是des
    		{
    			int end = i;
    			int des = arr[end + gap];
    			while (end >= 0)
    			{
    				if (des >= arr[end])
    					break;
    				if (des < arr[end])
    				{
    					arr[end + gap] = arr[end];
    					end -= gap;
    				}
    				arr[end + gap] = des;
    			}
    		}
     
    	}
    }
     
    //打印排序好的数组
    void PrintSort(int*arr,int n)
    {
    	for(int i=0;i					

    相关文章

    java速学教程(入门到精通)
    java速学教程(入门到精通)

    java怎么学习?java怎么入门?java在哪学?java怎么学才快?不用担心,这里为大家提供了java速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

    下载

    相关标签:

    本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

    相关专题

    更多
    Golang gRPC 服务开发与Protobuf实战
    Golang gRPC 服务开发与Protobuf实战

    本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

    8

    2026.01.15

    公务员递补名单公布时间 公务员递补要求
    公务员递补名单公布时间 公务员递补要求

    公务员递补名单公布时间不固定,通常在面试前,由招录单位(如国家知识产权局、海关等)发布,依据是原入围考生放弃资格,会按笔试成绩从高到低递补,递补考生需按公告要求限时确认并提交材料,及时参加面试/体检等后续环节。要求核心是按招录单位公告及时响应、提交材料(确认书、资格复审材料)并准时参加面试。

    44

    2026.01.15

    公务员调剂条件 2026调剂公告时间
    公务员调剂条件 2026调剂公告时间

    (一)符合拟调剂职位所要求的资格条件。 (二)公共科目笔试成绩同时达到拟调剂职位和原报考职位的合格分数线,且考试类别相同。 拟调剂职位设置了专业科目笔试条件的,专业科目笔试成绩还须同时达到合格分数线,且考试类别相同。 (三)未进入原报考职位面试人员名单。

    58

    2026.01.15

    国考成绩查询入口 国考分数公布时间2026
    国考成绩查询入口 国考分数公布时间2026

    笔试成绩查询入口已开通,考生可登录国家公务员局中央机关及其直属机构2026年度考试录用公务员专题网站http://bm.scs.gov.cn/pp/gkweb/core/web/ui/business/examResult/written_result.html,查询笔试成绩和合格分数线,点击“笔试成绩查询”按钮,凭借身份证及准考证进行查询。

    11

    2026.01.15

    Java 桌面应用开发(JavaFX 实战)
    Java 桌面应用开发(JavaFX 实战)

    本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

    65

    2026.01.14

    php与html混编教程大全
    php与html混编教程大全

    本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

    36

    2026.01.13

    PHP 高性能
    PHP 高性能

    本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

    75

    2026.01.13

    MySQL数据库报错常见问题及解决方法大全
    MySQL数据库报错常见问题及解决方法大全

    本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

    21

    2026.01.13

    PHP 文件上传
    PHP 文件上传

    本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

    35

    2026.01.13

    热门下载

    更多
    网站特效
    /
    网站源码
    /
    网站素材
    /
    前端模板

    精品课程

    更多
    相关推荐
    /
    热门推荐
    /
    最新课程
    Kotlin 教程
    Kotlin 教程

    共23课时 | 2.5万人学习

    C# 教程
    C# 教程

    共94课时 | 6.8万人学习

    Java 教程
    Java 教程

    共578课时 | 46.2万人学习

    关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
    php中文网:公益在线php培训,帮助PHP学习者快速成长!
    关注服务号 技术交流群
    PHP中文网订阅号
    每天精选资源文章推送

    Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号