希尔排序是一种看着像青铜的排序,当时当数据量很大时,他就是排序中的王者了。接下来慢慢回顾希尔排序
一、在了解希尔排序前我们先知道插入排序,因为希尔排序就是在插入排序的基础上进行优化的
先从单程入手:
![](https://i-blog.csdnimg.cn/blog_migrate/0ee86587b54969c6f64fb8a6eea22a0a.png)
这是排一个数时。
当我们需要排很多数时:
![](https://i-blog.csdnimg.cn/blog_migrate/0076c47669e1fe6a63543d1e0745ae52.png)
现在我们根据这个思路写出代码
void InserSort(int* a,int n)
{
for(int i = 0;i< n-1;i++)
{
int end = i;
int temp = a[end+1];
while(end>=0)
{
if(temp<a[end])
{
a[end+1] = a[end]
end--;
}
else
{
break;
}
}
a[end+1] = temp;
}
}
二、我们会了插入排序后,现在开始希尔排序
在插入排序时,如果当一个数是逆序时,插入排序的时间复杂度是O(N),所以希尔写了一个希尔排序来优化。
我们插入排序时每次间隔一位,现在我们间隔gap位,先进性预排序,预排序可以减少插入排序的时间,它先大致将比较大的数排到面去,小的排在前面。最后一次是插入排序进行排序,由于此前数据已经有很多被预排序排得有序了,所以最后一次的插入排序就很快了
![](https://i-blog.csdnimg.cn/blog_migrate/033de9fd583452b977ef7ff1b686f606.png)
首先先上手将思维图像画出来,
void ShellSort(int* a,int n)
{
//先将数据排gap组
while(gap>1)
{
gap = n/3 +1;//这里得确保最后一次gap等于1,否则会越界
for(int i = 0;i<n-gap;i++)
{
int end = i;
int temp = a[end+gap];
while(end>=0)
{
if(a[end]>temp)
{
a[end+gap] = a[end];
end -=gap;
}
else
break;
}
a[end+gap] = temp;
}
}
}
这是两者的对比图,单位值毫秒,排10万个数时,就已经差别那么大了
![](https://i-blog.csdnimg.cn/blog_migrate/6297f3bd6a88080f072a0181036c2907.png)
最后,希尔排序的时间复杂度,比较难算,目前使用O(N^1.3)