数据结构学习——shell排序的C语言实现

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语言,将其编写出来的。

 

 

  我们按照我们上述的原理将其一步一步的实现下:

     

 1 //函数的参数包含两个,一个是要排序的数组,一个是数组的大小
 2 //函数不使用额外的空间,只用到数组本身。这就要求数组在输入时候保留出
 3 //R[0],R[0]就相当是一个temp。用于互换两个数时的一个临时数。
 4 void shellsort(int R[],int n)
 5 {
 6     int i,j,d;
 7     d = n/2;
 8     while(d >= 1)
 9     {
10         //注意i的起始和最后的那个“=,我们的第一个数是R[1],所以i-d要从1开始
11         for(i = d + 1;i <= n;i++)
12         {
13             j = i - d;
14             //将R[i]暂存
15             R[0] = R[i];
16             //这边的R[0]就是R[j+d]也就是R[i]
17             while(j>0 && R[j] > R[0])
18             {
19                 //将j位置的值给j+d位置
20                 R[j+d] = R[j];
21                 j -= d;
22             }
23             //这边就是对应着上面的j-=d,如果上面的while执行了,j
24             //位置的值已经给了j+d位置了,那么j+d的值也要给j位置,互换
25             //这就要求下标必须是对应的。所以要将j-d才能满足这边是R[j]
26             //如果while没有执行,那么还将原来的R[i]的值放到原来位置上
27             R[j+d] = R[0];
28         }
29         //d减半
30         d/=2;
31     }
32 }

 

    上面就是按照所述的原理实现的,代码也不是很多,但是看上去有点乱,那么能不能改进点呢?下面就来了啊。

精简:

  我们一般在保证程序的可读上,将程序变得很精巧。这也是K&R的一个思想把。我们知道while和for在一定的情况下是可以替代的。所以我们可以将上述的程序,变得更为紧凑些,我们用三个for循环就可以了。下面是精简后的实现:

 

 1 #include<stdio.h>
 2 
 3 #define N 100
 4 void shellsort(int *,int);
 5 
 6 int main(void)
 7 {
 8     int v[N];
 9     int n,i;
10 
11     printf("输入你数组的大小:");
12     scanf("%d",&n);
13 
14     printf("输入%d个数,空格隔开:\n",n);
15     for(i = 0;i < n; i++)
16     {
17         scanf("%d",&v[i]);
18     }
19 
20     shellsort(v,n);
21     
22     printf("shell排序后:");
23     for(i =0;i < n;i++)
24     {
25         printf("%3d",v[i]);
26     }
27     
28     printf("\n");
29     return 0;
30 }
31 
32 void shellsort(int v[],int n)
33 {
34     int i,j,temp,gap;
35 
36     for(gap = n/2;gap > 0;gap /=2)
37         for(i =gap ;i < n ;i++)
38         for(j = i - gap;j>=0 && v[j] >v[j+gap]; j-=gap)
39          {
40             temp = v[j];
41             v[j] = v[j+gap];
42             v[j+gap] = temp;
43         }
44 }

 

这边我给出了一个完整的程序,不过我们只看函数的实现。上面的函数中,第一个for循环控制的是被比较数之间的间隔,也可以说是分组的依据。中间的循环是控制元素移动位置的。最后一个循环就是组内的比较,如果顺序不对就交换位置。将第一个函数里面的while整合成了for,这样看起来程序很精简,我们在理解上可能会有些卡壳,如果熟悉while和for之间转换的话就会好很多。

 

补充:

     

    for(表达式1;表达式2;表达式3)

      语句;

 

    等价于while的为:

 

    表达式1;

     while(表达式2)

  {

    语句;

    表达式3;

  }                

 

 

 

 

 

 

 

 

posted @ 2013-11-29 10:43  JesseEisen  阅读(1756)  评论(0编辑  收藏  举报