数据结构 ——— 希尔排序算法的实现

目录

前言

希尔排序算法的思想

第一步:先预排序

第二步:再直接插入排序

代码实现(默认排升序)

预排序思路的代码

希尔排序算法的代码实现


前言

在上一章学习了直接插入排序算法的实现

数据结构 --------- 直接插入排序算法的实现-CSDN博客

接下来要学习的是希尔排序,是在直接插入排序算法上优化的排序


希尔排序算法的思想

第一步:先预排序

预排序也就是把数组排成接近有序的状态

预排序的步骤:间隔为 gap 的长度分为一组,总计 gap 组,gap 的大小根据数组的长度改变
如上图所示,相同颜色的线条为一组,先对他们各自预排序

举例来说:9,6,3,1 为一组,那么就对这一组排序

排好后的结果就是 1,3,6,9 (默认升序)

其他各组和举例相同,最后预排序的结果如下

第二步:再直接插入排序

对数组预排序完成后,数据就是接近有序的状态了,这时再对数组进行直接插入排序即可

注意:第二步并不是调用直接插入排序函数,而是通过预排序的代码进行优化直接得出希尔排序,详情请见以下代码


代码实现(默认排升序)

预排序思路的代码

代码演示:

复制代码
int gap = 3; //待定

for (int i = 0; i < size - gap; i++)
{
	int end = i;
	int tmp = a[end + gap];

	while (end >= 0)
	{
		if (a[end] > tmp)
		{
			a[end + gap] = a[end];
			end -= gap;
		}
		else
		{
			break;
		}
	}

	a[end + gap] = tmp;
}

代码解析:

gap 的具体大小还需待定,先通过 a[] = { 9,8,7,6,5,4,3,3,2,1,0 },这组数据来举例说明

以上预排序代码的核心逻辑类似于直接插入排序算法

不同之处在于直接插入排序算法的 a[end] 和 tmp 间隔了 1

而预排序的 a[end] 和 tmp 间隔了 gap

这也就实现了相同颜色为一组,各自进行预排序

需要注意控制变量 i 的大小,防止越界

代码验证:

当 gap == 5 时:

当 gap == 3 时:

当 gap == 1 时:

结论:

当 gap 越大时,说明间隔越大,越不接近有序

当 gap 越小时,说明间隔越小,越接近有序

当 gap 为 1 时,就是直接插入排序算法,因为直接插入排序算法的 a[end] 和 tmp 的间隔就是 1

所以可以得出,想要将预排序直接优化成希尔排序,那么就要控制 gap 的大小

希尔排序算法的代码实现

代码演示:

复制代码
void ShellSort(int* a, int size)
{
	int gap = size;

	while (gap > 1)
	{
		gap = gap / 3 + 1;

		for (int i = 0; i < size - gap; i++)
		{
			int end = i;
			int tmp = a[end + gap];

			while (end >= 0)
			{
				if (a[end] > tmp)
				{
					a[end + gap] = a[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}

			a[end + gap] = tmp;
		}
	}
}

代码解析:

gap 初始值赋值为数组元素总个数的大小 size

每次循环 gap = gap / 3 + 1

+1 的作用是为了保证 gap 最后一次能被赋值为 1,因为 gap 为 1 时就是直接插入排序,这样做的作用避免了调用直接插入排序算法的函数,且在一个逻辑里面就对数组直接排好了序

当 gap == 1 时,进行预排序后,此时数组就是有序状态了,所以最外部的 while 循环判断 gap > 1 即可完成排序

代码验证:

相关推荐
DuHz5 小时前
论文精读:大语言模型 (Large Language Models, LLM) —— 一项调查
论文阅读·人工智能·深度学习·算法·机器学习·计算机视觉·语言模型
[J] 一坚5 小时前
嵌入式高手C
c语言·开发语言·stm32·单片机·mcu·51单片机·iot
加农炮手Jinx6 小时前
LeetCode 72. Edit Distance 题解
算法·leetcode·力扣
借雨醉东风6 小时前
程序分享--常见算法/编程面试题:旋转矩阵
c++·线性代数·算法·面试·职场和发展·矩阵
_深海凉_6 小时前
LeetCode热题100-打家劫舍
算法·leetcode·职场和发展
jghhh017 小时前
使用 MATLAB 实现支持向量回归 (SVR) 预测未来数据
算法·matlab
云泽8087 小时前
笔试算法 - 双指针篇(二):四大经典求和题型 + 有效三角形计数问题
c++·算法
qeen878 小时前
【数据结构】树的基本概念及存储
c语言·数据结构·c++·学习·
一江寒逸8 小时前
数据结构与算法之美:串(字符串)——从基础操作到KMP模式匹配,吃透面试最高频的字符串考点
数据结构·面试·职场和发展
刀法如飞8 小时前
【合并已排序数组的三种实现策略,哪一种更可取?】
算法·程序员