排序算法总结--希尔排序

希尔排序是一种改进的插入排序。前面我们讨论过插入排序在序列的大多数元素都是有序的情况下性能较好。所以希尔排序算法将原始的插入排序算法做了改进,每次比较不是相邻两个元素进行比较,而是隔几个元素之间进行比较,称为步长dk。通过不断缩小dk的值,逐步使得序列中元素有序。当dk缩小到1时,序列中的元素都已经基本有序,再进行插入排序就比较轻松了。

算法步骤:

1.选取合适的步长。可以选取元素个数的一半,也可以选1/3等等。

2.进行步长为dk的插入排序

3.缩小dk,进行递归

 

void Insort(int *a, int n, int dk) 
{
	int j = 0;
	for (int i = dk; i < n; ) 
	{
		j = i;
		while (a[j] < a[j - dk])
		{
			swap(&a[j], &a[j- dk]);
			j = j - dk;
			if (j < dk)
				break;
		}
		i = i + dk;
	}
	return;
}

void ShellSort(int *a, int n) {
	int dk = n/2;        //    设置初始dk
	while (dk >= 1) 
	{
		Insort(a, n, dk);
		dk /= 2;
	}
	return;
}

复杂度分析

时间复杂度分析:希尔排序的时间复杂度比较难分析,和序列本身以及步长的选择都有很大的关系,目前还没有个确定的值。查阅了基本数据结构的书籍以及网上的资料,众说纷纭。有两种主流的观点,一种认为是O(nlogn),一种则认为是O(N^3/2),总而言之希尔排序是一种较快的排序算法,速度介于冒泡和快排之间。但因为代码实现较为简单,在数据量较大的情况下也可以使用。

空间复杂度:只有递归消耗的栈空间。所以这个也和选择步长的计算方法有关。上面的例子空间复杂度就是O(logn).

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值