数据结构|排序算法(二) 希尔排序

一、希尔排序

1.算法思想

希尔排序(Shell Sort)是插入排序的一种改进版本,也称为缩小增量排序。

希尔排序的基本思想是将原始数据分成若干个子序列,每个子序列的元素间隔较大,然后对每个子序列进行插入排序。随着排序的进行,逐渐减小子序列的元素间隔,直到间隔为 1,此时整个序列基本有序,再进行一次普通的插入排序即可完成排序

注意:分组时采用间隔式的排序!

希尔排序采用间隔式排序主要有以下原因:

加快元素移动速度:希尔排序将原始数据分成多个子序列,每个子序列的元素间隔较大。这样在排序过程中,元素可以以较大的步长移动,能更快地将元素移动到其最终位置附近。例如,对于一个很大的数组,如果直接进行插入排序,每个元素可能需要移动很多次才能到达最终位置。但希尔排序通过大间隔分组,让元素先进行 "远距离" 的移动,初步将数据大致排序,减少了后续插入排序时元素的移动次数。

改善最坏时间复杂度:传统的插入排序在最坏情况下(如数组完全逆序)时间复杂度为O(n2)

希尔排序通过间隔式排序,使数组在初始阶段就接近有序状态,当间隔逐渐减小时,数组已经基本有序,此时再进行插入排序,就可以避免最坏情况的发生,其平均时间复杂度介于O(n)

到O(n2)之间,通常能达到O(n1.3)左右,性能相比直接插入排序有显著提升。

利用局部有序性:在间隔式排序过程中,每个子序列内部进行排序,使得子序列逐渐变得有序。随着间隔逐渐缩小,子序列的长度逐渐增加,而之前已经排好序的子序列能为后续更大规模的排序提供一定的有序基础,充分利用了数据的局部有序性,提高排序效率。

希尔排序核心思想就是:1 间隔式分组;2 利用直接插入排序使组内有序:越有序越快;3 再缩小分组再排序,直到缩为1组。

2.代码实现

复制代码
//希尔排序
//一趟希尔排序 gap为组数(间隔)
static void Shell(int* arr, int len, int gap)
{
	int tmp, i, j;
	for (i = gap; i < len; i++)//i是当前要处理的数字的下标
	{
		tmp = arr[i];//取出当前要插入的元素
		//遍历已排序的部分
		for (j = i - gap; j >= 0; j-=gap)//从i的前一个开始找位置,从后往前找,找比当前数字小的,同时移动数据
		{
			if (arr[j] > tmp)
			{
				arr[j + gap] = arr[j];//将比tmp大的元素后移gap
			}
			else
			{
				break;
			}
		}
		arr[j + gap] = tmp;//将tmp插入正确位置
	}
}
void ShellSort(int* arr, int len)
{
	int drr[] = { 5,3,1 };//分组数,最后一个组数一定为1
	for (int i = 0; i < sizeof(drr) / sizeof(drr[0]); i++)//循环控制分组的次数,此处分3次(5,3,1)
	{
		Shell(arr, len, drr[i]);//一趟希尔排序
	}
}

3.复杂度分析

希尔排序不稳定。

时间复杂度:希尔排序的时间复杂度与增量序列的选择有关。在最坏情况下,希尔排序的时间复杂度为O(n2)。在最好情况下,希尔排序的时间复杂度为O(nlogn)。

空间复杂度:希尔排序是一种原地排序算法,只需要常数级别的额外空间,因此空间复杂度为

O(1)。

稳定性:希尔排序是不稳定的排序算法。在排序过程中,相同元素的相对位置可能会发生改变。

希尔排序通过将原始数据分成多个子序列,每个子序列的元素间隔较大,使得元素可以更快地移动到其最终位置,从而提高了排序效率。它在处理大规模数据时表现较好,是一种常用的排序算法。

相关推荐
AI医影跨模态组学2 小时前
NPJ Precis Oncol(IF=8)复旦大学肿瘤医院等团队:基于生境CT放射组学解析可切除非小细胞肺癌时空异质性预测新辅助化疗免疫治疗病理反应
大数据·人工智能·算法·医学·医学影像
Book思议-3 小时前
二叉树的递归遍历详解:前序、中序与后序
数据结构·算法·二叉树的递归遍历-前中后序
Demon--hx3 小时前
[LeetCode]100 链表-专题
算法·leetcode·链表
Omics Pro3 小时前
首款多模态生物推理大语言模型
人工智能·算法·语言模型·自然语言处理·数据挖掘·数据分析·aigc
国产化创客3 小时前
基于ESP32+Wi‑Fi CSI的开源项目ESPectre
物联网·算法·信息与通信
_深海凉_3 小时前
LeetCode热题100-LRU 缓存
算法·leetcode·缓存
Via_Neo3 小时前
今天是周六,两天后是周几?
java·数据结构·算法
伟大的车尔尼3 小时前
广度优先搜索和深度优先搜索的概念
数据结构·算法·并查集·深度优先搜索·广度优先搜索