09-排序1 排序

给定N个(长整型范围内的)整数,要求输出从小到大排序后的结果。

本题旨在测试各种不同的排序算法在各种数据情况下的表现。各组测试数据特点如下:

数据1:只有1个元素;

数据2:11个不相同的整数,测试基本正确性;

数据3:103个随机整数;

数据4:104个随机整数;

数据5:105个随机整数;

数据6:105个顺序整数;

数据7:105个逆序整数;

数据8:105个基本有序的整数;

数据9:105个随机正整数,每个数字不超过1000。

输入格式:

输入第一行给出正整数N(≤10?^5??),随后一行给出N个(长整型范围内的)整数,其间以空格分隔。

输出格式:

在一行中输出从小到大排序后的结果,数字间以1个空格分隔,行末不得有多余空格。

输入样例:

11
4 981 10 -17 0 -20 29 50 8 43 -5

输出样例:

-20 -17 -5 0 4 8 10 29 43 50 981

总结:1.冒泡排序 插入排序 运行时间长,冒泡排序有运行超时。其他排序算法 运行时间差不多

2.希尔排序 堆排序 不稳定
3.归并排序 稳定 但需要额外开辟空间

  1 #include <stdio.h>
  2 #include <stdlib.h>
  3 typedef int ElementType;
  4
  5
  6 void BubbleSort(ElementType A[], int N)
  7 {
  8     for(int P = N-1; P >= 0; P--) {
  9         int flag = 0;
 10         for(int i = 0; i < P; i++) {
 11             if( A[i] > A[i+1] ) {
 12                 ElementType temp = A[i];    //swap A[i] A[i+1]
 13                 A[i] = A[i+1];
 14                 A[i+1] = temp;
 15                 flag = 1;
 16             }
 17         }
 18         if(flag == 0)
 19             break;
 20     }
 21 }
 22
 23 void InsertionSort(ElementType A[], int N)
 24 {
 25     int i;
 26     for (int P = 1; P < N; P++ ) {
 27         ElementType temp = A[P];     //取出未排序序列中的第一个元素
 28         for (i = P; i > 0 && A[i-1] > temp; i-- )
 29             A[i] = A[i-1];             //依次与已排序序列中元素比较并右移
 30         A[i] = temp;
 31     }
 32 }
 33
 34 /* 原始希尔排序 */
 35 void Shell_Sort(ElementType A[], int N)
 36 {
 37     int i;
 38     for(int D = N/2; D > 0; D/=2) {
 39         for(int P = D; P < N; P++) {
 40             ElementType temp = A[P];
 41             for(i = P; i >= D && A[i-D]>temp; i-=D)
 42                 A[i] = A[i-D];
 43             A[i] = temp;
 44         }
 45     }
 46 }
 47
 48 /* 希尔排序 - 用Sedgewick增量序列 */
 49 void ShellSort(ElementType A[], int N)
 50 {
 51      int Si, i;
 52      /* 这里只列出一小部分增量 */
 53      int Sedgewick[] = {929, 505, 209, 109, 41, 19, 5, 1, 0};
 54
 55      for ( Si = 0; Sedgewick[Si] >= N; Si++ )
 56          ; /* 初始的增量Sedgewick[Si]不能超过待排序列长度 */
 57
 58      for (int D = Sedgewick[Si]; D > 0; D = Sedgewick[++Si])
 59          for (int P = D; P < N; P++ ) {         //插入排序
 60              ElementType temp = A[P];
 61              for(i = P; i >= D && A[i-D]>temp; i-=D)
 62                  A[i] = A[i-D];
 63              A[i] = temp;
 64          }
 65 }
 66
 67 void Swap( ElementType *a, ElementType *b )
 68 {
 69     ElementType t = *a;
 70     *a = *b;
 71     *b = t;
 72 }
 73 /* 改编PercDown( MaxHeap H, int p )*/
 74 void PercDown( ElementType A[], int p, int N )
 75 {
 76   /* 将N个元素的数组中以A[p]为根的子堆调整为最大堆 */
 77     int Parent, Child;
 78
 79     ElementType X = A[p]; /* 取出根结点存放的值 */
 80     for( Parent=p; (Parent*2+1) < N; Parent=Child ) {
 81         Child = Parent * 2 + 1;
 82         if( (Child != N-1) && (A[Child] < A[Child+1]) )
 83             Child++;  /* Child指向左右子结点的较大者 */
 84         if( X >= A[Child] ) break; /* 找到了合适位置 */
 85         else  /* 下滤X */
 86             A[Parent] = A[Child];
 87     }
 88     A[Parent] = X;
 89 }
 90
 91 void HeapSort(ElementType A[], int N)
 92 {
 93      for(int i = N/2-1; i >= 0; i--)/* 建立最大堆 */
 94          PercDown( A, i, N );
 95
 96      for(int i = N-1; i > 0; i--) {
 97          /* 删除最大堆顶 */
 98          Swap(&A[0], &A[i] );
 99          PercDown(A, 0, i);
100      }
101 }
102
103 /* 归并排序 - 递归实现 */
104
105 /* L = 左边起始位置, R = 右边起始位置, RightEnd = 右边终点位置*/
106 void Merge( ElementType A[], ElementType TmpA[], int L, int R, int RightEnd )
107 { /* 将有序的A[L]~A[R-1]和A[R]~A[RightEnd]归并成一个有序序列 */
108
109      int LeftEnd = R - 1; /* 左边终点位置 */
110      int temp = L;         /* 有序序列的起始位置 */
111      int NumElements = RightEnd - L + 1;
112
113      while( L <= LeftEnd && R <= RightEnd ) {
114          if ( A[L] <= A[R] )
115              TmpA[temp++] = A[L++]; /* 将左边元素复制到TmpA */
116          else
117              TmpA[temp++] = A[R++]; /* 将右边元素复制到TmpA */
118      }
119
120      while( L <= LeftEnd )
121          TmpA[temp++] = A[L++]; /* 直接复制左边剩下的 */
122      while( R <= RightEnd )
123          TmpA[temp++] = A[R++]; /* 直接复制右边剩下的 */
124
125      for(int i = 0; i < NumElements; i++, RightEnd -- )
126          A[RightEnd] = TmpA[RightEnd]; /* 将有序的TmpA[]复制回A[] */
127 }
128 /* 核心递归排序函数 */
129 void Msort( ElementType A[], ElementType TmpA[], int L, int RightEnd )
130 {
131      int Center;
132
133      if ( L < RightEnd ) {
134           Center = (L+RightEnd) / 2;
135           Msort( A, TmpA, L, Center );              /* 递归解决左边 */
136           Msort( A, TmpA, Center+1, RightEnd );     /* 递归解决右边 */
137           Merge( A, TmpA, L, Center+1, RightEnd );  /* 合并两段有序序列 */
138      }
139 }
140 /* 归并排序接口函数 */
141 void MergeSort( ElementType A[], int N )
142 {
143      ElementType *TmpA;
144      TmpA = (ElementType *)malloc(N*sizeof(ElementType));
145
146      if ( TmpA != NULL ) {
147           Msort( A, TmpA, 0, N-1 );
148           free( TmpA );
149      }
150      else printf( "空间不足" );
151 }
152
153 /* 归并排序 - 循环实现 */
154
155 /* 这里Merge函数在递归版本中给出 */
156 /* length = 当前有序子列的长度*/
157 void Merge_pass( ElementType A[], ElementType TmpA[], int N, int length )
158 { /* 两两归并相邻有序子列 */
159      int i, j;
160
161      for ( i = 0; i <= N-2*length; i += 2*length )
162          Merge( A, TmpA, i, i+length, i+2*length-1 );
163      if ( i+length < N ) /* 归并最后2个子列*/
164          Merge( A, TmpA, i, i+length, N-1);
165      else /* 最后只剩1个子列*/
166          for ( j = i; j < N; j++ ) TmpA[j] = A[j];
167 }
168
169 void Merge_Sort( ElementType A[], int N )
170 {
171      int length;
172      ElementType *TmpA;
173
174      length = 1; /* 初始化子序列长度*/
175      TmpA = (ElementType *)malloc( N * sizeof( ElementType ) );
176      if ( TmpA != NULL ) {
177           while( length < N ) {
178               Merge_pass( A, TmpA, N, length );
179               length *= 2;
180               Merge_pass( TmpA, A, N, length );
181               length *= 2;
182           }
183           free( TmpA );
184      }
185      else printf( "空间不足" );
186 }
187
188
189
190 int main()
191 {
192     int N;
193     int a[100005];
194     scanf("%d",&N);
195     for(int i = 0; i < N; i++)
196         scanf("%d",&a[i]);
197 //    BubbleSort(a,N);
198 //    InsertionSort(a,N);
199 //    Shell_Sort(a,N);
200 //    ShellSort(a,N);    //希尔排序 - 用Sedgewick增量序列
201 //    HeapSort(a,N);
202 //  MergeSort(a,N);    //归并排序 - 递归实现
203 //    Merge_Sort(a,N);
204     for(int i = 0; i < N-1; i++)
205         printf("%d ",a[i]);
206     printf("%d\n",a[N-1]);
207     return 0;
208 } 







时间: 2024-10-18 03:59:05

09-排序1 排序的相关文章

C# 插入排序 冒泡排序 选择排序 高速排序 堆排序 归并排序 基数排序 希尔排序

以下列出了数据结构与算法的八种基本排序:插入排序 冒泡排序 选择排序 高速排序 堆排序 归并排序 基数排序 希尔排序,然后是測试的样例.代码位置:http://download.csdn.net/detail/luozuolincool/8040027 排序类: public class Sortings { //插入排序 public void insertSort(int[] array) { int temp = 0; int index = 0; for (int i = 0; i <

java-数组排序--计数排序、桶排序、基数排序

计数排序引入 不难发现不论是冒泡排序还是插入排序,其排序方法都是通过对每一个数进行两两比较进行排序的,这种方法称为比较排序,实际上对每个数的两两比较严重影响了其效率,理论上比较排序时间复杂度的最低下限为nlog(n),即任何比较排序的时间复杂度将不会低于nlog(n),那么有没有方法能不经过数列比较就能使数列排序呢 ,她们的时间复杂度又是多少呢??? 计数排序就是一个非比较排序的算法,一如鱼与熊掌不可兼得,她使用了牺牲空间换时间的方法,使的时间复杂度可以达到Ο(n+k) 假设我们有一个数列arr

给object数组进行排序(排序条件是每个元素对象的属性个数)

从汤姆大叔的博客里看到了6个基础题目:本篇是第3题 - 给object数组进行排序(排序条件是每个元素对象的属性个数) 解题关键: 1.Array.sort的用法 2.object的属性数量的统计 解点1:Array.sort的用法 Array.sort可以为数组指定一个排序规则,一般用如下格式进行指定,代码如下: var arr = [10,6,0,4]; console.log( arr.sort() ); //按字符排序 0,10,4,6 console.log( arr.sort( fu

HDU 1862 EXCEL排序 (排序水题)

Problem Description Excel可以对一组纪录按任意指定列排序.现请你编写程序实现类似功能. Input 测试输入包含若干测试用例.每个测试用例的第1行包含两个整数 N (<=100000) 和 C,其中 N 是纪录的条数,C 是指定排序的列号.以下有 N 行,每行包含一条学生纪录.每条学生纪录由学号(6位数字,同组测试中没有重复的学号).姓名(不超过8位且不包含空格的字符串).成绩(闭区间[0, 100]内的整数)组成,每个项目间用1个空格隔开.当读到 N=0 时,全部输入结

算法大神之路----排序(选择排序法)

选择排序法,顾名思义,就是把特定的数据选择出来进行排序. 选择排序法有两种方式 在所有的数据中,当由大到小排序,那么就将最大值放到第一个位置 如果由小到大排序,那么就将最小值放到第一个位置 以由小到大排序举例,当排序时候,扫描整个数据,拿第一个依次与其他做比较,如果其他数据比第一个大,或者相等,那么就不交换,如果其他数据比第一个数小,那么就交换二者的位置,扫描结束后,则从第二个数开始,依次扫描. 方法分析 无论是最坏还是最好情况,甚至是平均情况下,都需要对全部数据进行扫描,找到最大或最小值,因此

排序——选择排序

在要排序的一组数中,选出最小(或者最大)的一个数与第1个位置的数交换:然后在剩下的数其中再找最小(或者最大)的与第2个位置的数交换,依次类推,直到第n-1个元素(倒数第二个数)和第n个元素(最后一个数)比較为止. 程序流程: 第一趟,从n 个记录中找出关键码最小的记录与第一个记录交换: 第二趟,从第二个记录開始的n-1 个记录中再选出关键码最小的记录与第二个记录交换: 以此类推..... 第i 趟,则从第i 个记录開始的n-i+1 个记录中选出关键码最小的记录与第i 个记录交换,直到整个序列按关

排序算法3--插入排序--希尔排序(缩小增量排序)

希尔排序(缩小增量排序) 希尔排序(Shell Sort)是插入排序的一种.也称缩小增量排序,是直接插入排序算法的一种更高效的改进版本. 该方法的基本思想是:先将整个待排元素序列分割成若干个子序列(由相隔某个“增量”的元素组成的)分别进行直接插入排序,然后依次缩减增量再进行排序,待整个序列中的元素基本有序(增量足够小)时,再对全体元素进行一次直接插入排序.因为直接插入排序在元素基本有序的情况下(接近最好情况),效率是很高的,因此希尔排序在时间效率上比前两种方法有较大提高. 以n=10的一个数组4

10-6-起泡排序-内部排序-第10章-《数据结构》课本源码-严蔚敏吴伟民版

课本源码部分 第10章  内部排序 - 起泡排序 ——<数据结构>-严蔚敏.吴伟民版        源码使用说明  链接??? <数据结构-C语言版>(严蔚敏,吴伟民版)课本源码+习题集解析使用说明        课本源码合辑  链接??? <数据结构>课本源码合辑        习题集全解析  链接??? <数据结构题集>习题解析合辑        本源码引入的文件  链接? SequenceListType.c        相关测试数据下载  链接?

10-5-希尔排序-内部排序-第10章-《数据结构》课本源码-严蔚敏吴伟民版

课本源码部分 第10章  内部排序 - 希尔排序 ——<数据结构>-严蔚敏.吴伟民版        源码使用说明  链接??? <数据结构-C语言版>(严蔚敏,吴伟民版)课本源码+习题集解析使用说明        课本源码合辑  链接??? <数据结构>课本源码合辑        习题集全解析  链接??? <数据结构题集>习题解析合辑        本源码引入的文件  链接? SequenceListType.c        相关测试数据下载  链接?

【啊哈!算法】最快最简单的排序——桶排序

转自:http://bbs.ahalei.com/thread-4399-1-1.html 最快最简单的排序——桶排序 在我们生活的这个世界中到处都是被排序过的.站队的时候会按照身高排序,考试的名次需要按照分数排序,网上购物的时候会按照价格排序,电子邮箱中的邮件按照时间排序……总之很多东西都需要排序,可以说排序是无处不在.现在我们举个具体的例子来介绍一下排序算法. 首先出场的我们的主人公小哼,上面这个可爱的娃就是啦.期末考试完了老师要将同学们的分数按照从高到低排序.小哼的班上只有5个同学,这5个