shell排序:
这个排序的命名是来自发明者的名字,和排序的方法没有字面上的联系。所以不要因为名字而感觉很难。在K&R的C程序设计语言中书中只用了几行代码很简洁的实现了这个排序算法。那就来看看这个排序是如何实现的。
原理:
我们将所要排序的序列(大小为n)划分成组,组的数量一般是可以用这个序列的大小的一半来定义(也就是d = n/2),然后不断折半,而组的成员就是间隔为d的数分为一组。比如这边有个长度为8的数字序列要去排序,那我们就可以先将这个序列分成d=4组的,每个组有两个数,(这边的4就是8的一半)。这四组就是(R1,R5),(R2,R6),(R3,R7),(R4,R8).然后就是组内比较,如果前者大,交换他们的位置。以次类推。编写程序时就是将我们所取的数和与他间隔为d大小的位置上的数比较。这边第一次组的数量是4,那么就是将第一个数和他间隔为4的数比较。当第一次排序后,我们将这个间隔缩小,就是拿上一次组数d折半。那么这次的组就只有2个了。每个组有4个数.同样是和上面一样,间隔为d=2的位置上的数比较(实际上操作是这样的,但是我们可以认为他们已经是在一个组里面了,就可以说成组内比较)。我们知道d最后会变成1的,那会儿就是相邻的两个数比较,所有的数字都会放在正确的位置上的。原理基本是这样的。
实现:
我们知道了原理,那就用自己熟悉的语言将其编写出来,我们这边是用的C语言,将其编写出来的。
我们按照我们上述的原理将其一步一步的实现下:
//函数的参数包含两个,一个是要排序的数组,一个是数组的大小
//函数不使用额外的空间,只用到数组本身。这就要求数组在输入时候保留出
//R[0],R[0]就相当是一个temp。用于互换两个数时的一个临时数。
void shellsort(int R[],int n)
{
int i,j,d;
d = n/;
while(d >= )
{
//注意i的起始和最后的那个“=,我们的第一个数是R[1],所以i-d要从1开始
for(i = d + ;i <= n;i++)
{
j = i - d;
//将R[i]暂存
R[] = R[i];
//这边的R[0]就是R[j+d]也就是R[i]
while(j> && R[j] > R[])
{
//将j位置的值给j+d位置
R[j+d] = R[j];
j -= d;
}
//这边就是对应着上面的j-=d,如果上面的while执行了,j
//位置的值已经给了j+d位置了,那么j+d的值也要给j位置,互换
//这就要求下标必须是对应的。所以要将j-d才能满足这边是R[j]
//如果while没有执行,那么还将原来的R[i]的值放到原来位置上
R[j+d] = R[];
}
//d减半
d/=;
}
}
上面就是按照所述的原理实现的,代码也不是很多,但是看上去有点乱,那么能不能改进点呢?下面就来了啊。
精简:
我们一般在保证程序的可读上,将程序变得很精巧。这也是K&R的一个思想把。我们知道while和for在一定的情况下是可以替代的。所以我们可以将上述的程序,变得更为紧凑些,我们用三个for循环就可以了。下面是精简后的实现:
#include<stdio.h> #define N 100
void shellsort(int *,int); int main(void)
{
int v[N];
int n,i; printf("输入你数组的大小:");
scanf("%d",&n); printf("输入%d个数,空格隔开:\n",n);
for(i = ;i < n; i++)
{
scanf("%d",&v[i]);
} shellsort(v,n); printf("shell排序后:");
for(i =;i < n;i++)
{
printf("%3d",v[i]);
} printf("\n");
return ;
} void shellsort(int v[],int n)
{
int i,j,temp,gap; for(gap = n/;gap > ;gap /=)
for(i =gap ;i < n ;i++)
for(j = i - gap;j>= && v[j] >v[j+gap]; j-=gap)
{
temp = v[j];
v[j] = v[j+gap];
v[j+gap] = temp;
}
}
这边我给出了一个完整的程序,不过我们只看函数的实现。上面的函数中,第一个for循环控制的是被比较数之间的间隔,也可以说是分组的依据。中间的循环是控制元素移动位置的。最后一个循环就是组内的比较,如果顺序不对就交换位置。将第一个函数里面的while整合成了for,这样看起来程序很精简,我们在理解上可能会有些卡壳,如果熟悉while和for之间转换的话就会好很多。
补充:
for(表达式1;表达式2;表达式3)
语句;
等价于while的为:
表达式1;
while(表达式2)
{
语句;
表达式3;
}