常见的六种排序算法实现
常见的六种排序算法实现
在写六种排序先写两个辅助的函数
交换数组的元素函数swap
/*辅助交换函数*/ int Swap(int* str, int pos1, int pos2) { if (str == NULL) { return -1; } if (pos1 < 0 || pos2 < 0) { return -2; } /*交换数组pos1的值和pos2的值*/ int temp = str[pos1]; str[pos1] = str[pos2]; str[pos2] = temp; return 0; } /*打印数组元素*/ void PrintArraryElement(int* str, int length) { assert(str != NULL); printf("\t\t\t"); for (int i = 0; i < length; i++) { printf("%d\t", str[i]); } printf("\n"); }
选择排序
/*选择排序*/ int SelectSort(int* str, int length) { /* 算法思想 第一次:i=0(j=1,此时比较str[0],str[1]。如果str[1]比str[0]还小的话,进行交换,此时str[0]是前2个最小值了。接着j=2的时候,也就是第3个元素的时候,在进行比较str[0],str[2]。如果str[2]比str[0]小的话,进行交换。此时str[0]就是前3个最小值了。类推。i=0;执行完的时候第一个元素str[0]一定是所有元素中最小的那一个。这样第一个元素就是有序的。 第二次:i=1执行过程和i=0类似,只是每次都是和str[1]进行比较了。str[0]是不参与比较的。所以i=1执行完后,str[1]一定是str[1]到str[length-1] 元素的最小值。这样前2个元素是有序的。 第三次:i=2。这样前3个元素是有序的。 第N次:i=n-1。 这样前N个元素是有序的。也就是所有元素都是有序的啦。 总结:每次都是和固定的位置那个元素进行比较。 */ assert(str != NULL); for (int i = 0; i < length; i++) { for (int j = i + 1; j < length; j++) { if (str[i] >str[j]) { Swap(str, i, j); } } } return 0; }
冒泡排序
/*冒泡排序*/ int BubbleSort(int* str, int length) { /* 算法思想 第一次:i=0(j=0,此时比较str[0],str[1]。如果str[1]比str[0]还小的话,进行交换,此时str[0]是前2个最小值了 str[1]是前2个最大值。接着j=1的时候,在进行比较str[1],str[2]。如果str[2]比str[1]小的话,进行交换。此时str[2]就是前3个最大值了。类推。i=0;执行完的时候最后一个元素str[length-1]一定是所有元素中最大的那一个。这样最后一个元素就是有序的。 第二次:i=1执行过程和i=0类似,每次相邻元素进行比较。所以i=1执行完后,倒数第二个元素str[length-2]一定是str[0]到str[length-2] 元素的最小值。这样后2个元素是有序的。 第三次:i=2。这样后3个元素是有序的。 第N次:i=n-1。 这样后N个元素是有序的。也就是所有元素都是有序的啦。 总结:每次都是相邻的元素进行比较,可以定义一个flag标示变量前面的结果是否有序进行优化。 */ assert(str != NULL); for (int i = 0; i < length; i++) { for (int j = 0; j < length - 1; j++) { if (str[j] > str[j + 1]) { Swap(str, j, j + 1); } } } return 0; }
插入排序
/*插入排序*/ int InsertSort(int* str, int length) { /* 算法思想 这里举例个数据 1 , 2 ,4 , 5 , 3 核心:就是取出来一个元素,然后和前面的有序序列进行比较,找到合适的位置。插进去,这个位置的后面的需要后移。 对于上面的这个数组一共5个元素,前4个是有序的。那么怎么讲第5个元素(3)插入到指定位置呢。 1.取出来: 先用一个变量pos 保存3这个元素的位置下标(pos =4),用一个value保存这个元素的值(3 )。 2.进行比较: 前面的4个元素(1,2,4,5)是有序的。先将 3和有序序列的最后一个元素(5)进行比较,3是小于5的,那么就将5这个元素放到原来3的为位置上去,这样5的这个位置就空出来了(也就是取出来了 pos =3)在和4进行比较。3小于4的。那么4这个元素后移下。这样pos=2的位置就空出来了。 3在和2进行比较,3是大于2的,for循环结束 把value(3) 赋值到pos位置上去,也就是str[pos] =value,级str[2] =3; 3.这样结果就是有效的了。 当然开始的时候前面几个不是有序的,至少第一个元素是有序的。那么就从第二个元素开始进行插入。也就是i=1开始,i=1结束的时候前面2个元素是有序 i=2,结束的时候前面3个元素是有序的。i=length -1,结束的时候前面length 个元素都是有序的。也就是整个数组是有序的。 */ assert(str != NULL); for (int i = 1; i < length; i++) { int pos = i; int value = str[i]; for (int j = i-1; j >=0 && str[j] >value; j--) { str[j + 1] = str[j]; pos = j; } str[pos] = value; } return 0; }
希尔排序
/*希尔排序*/ int ShellSort(int* str, int length) { /* 算法思想: 核心代码和插入排序一样。加入了分组处理,gap=gap/3+1 这样分组是效率最高的。 */ assert(str != NULL); int gap = length; do { gap = gap / 3 + 1; for (int i = gap; i < length; i+=gap) { int pos = i; int value = str[i]; for (int j = i - gap; j >= 0 && str[j] >value; j-=gap) { str[j + gap] = str[j]; pos = j; } str[pos] = value; } } while (gap > 1); return 0; }
快速排序
/*快速排序*/ /*******************************************************************************/ int Partition(int* str, int low, int high) { /* 函数功能:就是分隔str的low元素到high的元素(一共high-low+1个)比第一个元素(temp=str[low])大的都放到右边,比temp小的都放到左边。 算法思想: 这里给一个简单的数组来解释这个函数的功能 str 的数组元素(7个 low =0 high =6) 8,7,2,4,8,9,2 1.取temp =str[low]=8; 2;low就是第一个元素的下标,high就是最后一个元素的下标。第一次判断str[high] >= temp 为假,交换8和2 此时low =0 high =6数组变成了2,7,2,4,8,9,8 接着判断str[low]<=temp(2<8)为真,low++,此时low =1,high =6 ,接着判断str[low]<=temp(7<8)为真,low++,此时low=2,high=6,接着判断str[low]<=temp(2<8)为真low++,此时low=3,high=6 接着判断str[low]<=temp(4<8)为真low++,此时low=4,high=6,接着判断str[low]<=temp(8<=8)为真low++,low=5,high=6,接着判断str[low]<=temp(9<=8)为假,交换9和8,此时low=5,high=6 数组变成了2,7,2,4,8,8,9,接着进行判断str[high]>=temp(9>=8)为真,high--此时low=5,high=5 ,交换low和high位置元素,其实是一样的。最终返回一个low也就是5,也就是初始的第一个元素现在在这个数组的位置下标啦。最终得到的数组就是2,7,2,4,8,8,9. */ assert(str != NULL); int temp = str[low]; while (low <high) { while (low <high && str[high] >= temp) { high--; } Swap(str, low, high); while (low <high && str[low] <= temp) { low++; } Swap(str, low, high); } return low; } int QSort(int* str, int low, int high) { /* 算法思想: int par =Partition(str,low,high)执行后返回一个数组下标值par,str数组中数组下标小于par的元素都是小于str[par]的。数组下标大于par的元素都是大于str[par]的。这个时候str[par]就是不会在改变了。 QSort(str, low, par - 1);就是给low 到par-1的那些元素进行分割,每次都会确定一个str[par]的值不会再去改变的。 剩下的就是递归调用完成最终的排序计算。 */ assert(str != NULL); if (low < high) { int par = Partition(str, low, high); QSort(str, low, par - 1); QSort(str, par + 1, high); } return 0; } int QuickSort(int* str, int length) { assert(str != NULL); return QSort(str, 0, length - 1); }
归并排序
/*归并排序*/ /*******************************************************************************/ void Merge(int* src, int* dst, int low, int mid, int high) { /* 归并排序算法思想: src1 low到mid是有序的,mid到high是有序的。 这个函数就是将2个各自有序的数组合成一个数组 一个数组是1 ,4, 5 ,7 一个数组是2,3 ,6,8,9 10 第一次判断src[i] <src[j] 即1<2为真 dst[k++] =1;将1赋值到结果数组去i++,k++ 第二次判断src[i] <src[j}}即2<2为假 dst[k++] =2;将2赋值到结果数组去 j++,k++ 最终会有一个数组剩下元素,另一个数组没有元素的。 将那个数组的元素copy到结果数组中去。 */ assert(src != NULL); assert(dst != NULL); int i = low; int j = mid + 1; int k = low; while (i<=mid && j<=high) { /*每次插入两个数组的头部最小值到目标数组中,保证了目标数组是有序的*/ if (src[i] < src[j]) { dst[k++] = src[i++]; } else { dst[k++] = src[j++]; } } /*将剩下的那部分copy到目标数组中去*/ while (i<=mid) { dst[k++] = src[i++]; } while (j<=high) { dst[k++] = src[j++]; } } int MSort(int* src, int* dst, int low, int high, int max) { assert(src != NULL); assert(dst != NULL); if (low == high) { /*只有一个元素的时候直接给目标数组赋值了。不用在递归啦*/ dst[low] = src[low]; return 0; } else { int mid = (low + high) / 2; int *space = (int*)malloc(sizeof(int)* max); if (space == NULL) { return -1; } //递归调用进行分割直到只有一个元素递归结束 MSort(src, space, low, mid, max); MSort(src, space, mid + 1, high, max); //对递归的数据进行合并 Merge(space, dst, low, mid, high); if (space != NULL) { free(space); } return 0; } return 0; } int MergeSort(int* str,int length) { assert(str != NULL); return MSort(str, str, 0, length - 1, length); }
以上代码汇总
sort.h文件
#include <stdio.h> #include <stdlib.h> #include <string.h> #ifndef _SORT_H_ #define _SORT_H_ #ifdef __cplusplus extern "C" { #endif int __declspec(dllexport) SelectSort(int* str, int length); void __declspec(dllexport) PrintArraryElement(int* str, int length); int __declspec(dllexport) BubbleSort(int* str, int count); int __declspec(dllexport) InsertSort(int* str, int length); int __declspec(dllexport) ShellSort(int* str, int length); int __declspec(dllexport) QSort(int* str, int low, int high); int __declspec(dllexport) QuickSort(int* str, int length); int __declspec(dllexport) MergeSort(int* str, int length); #ifdef __cplusplus } #endif #endif
sort.cpp文件
#include <stdio.h> #include <stdlib.h> #include <string.h> #include "Sort.h" #include <assert.h> /*辅助交换函数*/ int Swap(int* str, int pos1, int pos2) { if (str == NULL) { return -1; } if (pos1 < 0 || pos2 < 0) { return -2; } /*交换数组pos1的值和pos2的值*/ int temp = str[pos1]; str[pos1] = str[pos2]; str[pos2] = temp; return 0; } /*打印数组元素*/ void PrintArraryElement(int* str, int length) { assert(str != NULL); printf("\t\t\t"); for (int i = 0; i < length; i++) { printf("%d\t", str[i]); } printf("\n"); } /*选择排序*/ int SelectSort(int* str, int length) { /* 算法思想 第一次:i=0(j=1,此时比较str[0],str[1]。如果str[1]比str[0]还小的话,进行交换,此时str[0]是前2个最小值了。接着j=2的时候,也就是第3个元素的时候,在进行比较str[0],str[2]。如果str[2]比str[0]小的话,进行交换。此时str[0]就是前3个最小值了。类推。i=0;执行完的时候第一个元素str[0]一定是所有元素中最小的那一个。这样第一个元素就是有序的。 第二次:i=1执行过程和i=0类似,只是每次都是和str[1]进行比较了。str[0]是不参与比较的。所以i=1执行完后,str[1]一定是str[1]到str[length-1] 元素的最小值。这样前2个元素是有序的。 第三次:i=2。这样前3个元素是有序的。 第N次:i=n-1。 这样前N个元素是有序的。也就是所有元素都是有序的啦。 总结:每次都是和固定的位置那个元素进行比较。 */ assert(str != NULL); for (int i = 0; i < length; i++) { for (int j = i + 1; j < length; j++) { if (str[i] >str[j]) { Swap(str, i, j); } } } return 0; } /*冒泡排序*/ int BubbleSort(int* str, int length) { /* 算法思想 第一次:i=0(j=0,此时比较str[0],str[1]。如果str[1]比str[0]还小的话,进行交换,此时str[0]是前2个最小值了 str[1]是前2个最大值。接着j=1的时候,在进行比较str[1],str[2]。如果str[2]比str[1]小的话,进行交换。此时str[2]就是前3个最大值了。类推。i=0;执行完的时候最后一个元素str[length-1]一定是所有元素中最大的那一个。这样最后一个元素就是有序的。 第二次:i=1执行过程和i=0类似,每次相邻元素进行比较。所以i=1执行完后,倒数第二个元素str[length-2]一定是str[0]到str[length-2] 元素的最小值。这样后2个元素是有序的。 第三次:i=2。这样后3个元素是有序的。 第N次:i=n-1。 这样后N个元素是有序的。也就是所有元素都是有序的啦。 总结:每次都是相邻的元素进行比较,可以定义一个flag标示变量前面的结果是否有序进行优化。 */ assert(str != NULL); for (int i = 0; i < length; i++) { for (int j = 0; j < length - 1; j++) { if (str[j] > str[j + 1]) { Swap(str, j, j + 1); } } } return 0; } /*插入排序*/ int InsertSort(int* str, int length) { /* 算法思想 这里举例个数据 1 , 2 ,4 , 5 , 3 核心:就是取出来一个元素,然后和前面的有序序列进行比较,找到合适的位置。插进去,这个位置的后面的需要后移。 对于上面的这个数组一共5个元素,前4个是有序的。那么怎么讲第5个元素(3)插入到指定位置呢。 1.取出来: 先用一个变量pos 保存3这个元素的位置下标(pos =4),用一个value保存这个元素的值(3 )。 2.进行比较: 前面的4个元素(1,2,4,5)是有序的。先将 3和有序序列的最后一个元素(5)进行比较,3是小于5的,那么就将5这个元素放到原来3的为位置上去,这样5的这个位置就空出来了(也就是取出来了 pos =3)在和4进行比较。3小于4的。那么4这个元素后移下。这样pos=2的位置就空出来了。 3在和2进行比较,3是大于2的,for循环结束 把value(3) 赋值到pos位置上去,也就是str[pos] =value,级str[2] =3; 3.这样结果就是有效的了。 当然开始的时候前面几个不是有序的,至少第一个元素是有序的。那么就从第二个元素开始进行插入。也就是i=1开始,i=1结束的时候前面2个元素是有序 i=2,结束的时候前面3个元素是有序的。i=length -1,结束的时候前面length 个元素都是有序的。也就是整个数组是有序的。 */ assert(str != NULL); for (int i = 1; i < length; i++) { int pos = i; int value = str[i]; for (int j = i-1; j >=0 && str[j] >value; j--) { str[j + 1] = str[j]; pos = j; } str[pos] = value; } return 0; } /*希尔排序*/ int ShellSort(int* str, int length) { /* 算法思想: 核心代码和插入排序一样。加入了分组处理,gap=gap/3+1 这样分组是效率最高的。 */ assert(str != NULL); int gap = length; do { gap = gap / 3 + 1; for (int i = gap; i < length; i+=gap) { int pos = i; int value = str[i]; for (int j = i - gap; j >= 0 && str[j] >value; j-=gap) { str[j + gap] = str[j]; pos = j; } str[pos] = value; } } while (gap > 1); return 0; } /*快速排序*/ /*******************************************************************************/ int Partition(int* str, int low, int high) { /* 函数功能:就是分隔str的low元素到high的元素(一共high-low+1个)比第一个元素(temp=str[low])大的都放到右边,比temp小的都放到左边。 算法思想: 这里给一个简单的数组来解释这个函数的功能 str 的数组元素(7个 low =0 high =6) 8,7,2,4,8,9,2 1.取temp =str[low]=8; 2;low就是第一个元素的下标,high就是最后一个元素的下标。第一次判断str[high] >= temp 为假,交换8和2 此时low =0 high =6数组变成了2,7,2,4,8,9,8 接着判断str[low]<=temp(2<8)为真,low++,此时low =1,high =6 ,接着判断str[low]<=temp(7<8)为真,low++,此时low=2,high=6,接着判断str[low]<=temp(2<8)为真low++,此时low=3,high=6 接着判断str[low]<=temp(4<8)为真low++,此时low=4,high=6,接着判断str[low]<=temp(8<=8)为真low++,low=5,high=6,接着判断str[low]<=temp(9<=8)为假,交换9和8,此时low=5,high=6 数组变成了2,7,2,4,8,8,9,接着进行判断str[high]>=temp(9>=8)为真,high--此时low=5,high=5 ,交换low和high位置元素,其实是一样的。最终返回一个low也就是5,也就是初始的第一个元素现在在这个数组的位置下标啦。最终得到的数组就是2,7,2,4,8,8,9. */ assert(str != NULL); int temp = str[low]; while (low <high) { while (low <high && str[high] >= temp) { high--; } Swap(str, low, high); while (low <high && str[low] <= temp) { low++; } Swap(str, low, high); } return low; } int QSort(int* str, int low, int high) { /* 算法思想: int par =Partition(str,low,high)执行后返回一个数组下标值par,str数组中数组下标小于par的元素都是小于str[par]的。数组下标大于par的元素都是大于str[par]的。这个时候str[par]就是不会在改变了。 QSort(str, low, par - 1);就是给low 到par-1的那些元素进行分割,每次都会确定一个str[par]的值不会再去改变的。 剩下的就是递归调用完成最终的排序计算。 */ assert(str != NULL); if (low < high) { int par = Partition(str, low, high); QSort(str, low, par - 1); QSort(str, par + 1, high); } return 0; } int QuickSort(int* str, int length) { assert(str != NULL); return QSort(str, 0, length - 1); } /*归并排序*/ /*******************************************************************************/ void Merge(int* src, int* dst, int low, int mid, int high) { /* 归并排序算法思想: src1 low到mid是有序的,mid到high是有序的。 这个函数就是将2个各自有序的数组合成一个数组 一个数组是1 ,4, 5 ,7 一个数组是2,3 ,6,8,9 10 第一次判断src[i] <src[j] 即1<2为真 dst[k++] =1;将1赋值到结果数组去i++,k++ 第二次判断src[i] <src[j}}即2<2为假 dst[k++] =2;将2赋值到结果数组去 j++,k++ 最终会有一个数组剩下元素,另一个数组没有元素的。 将那个数组的元素copy到结果数组中去。 */ assert(src != NULL); assert(dst != NULL); int i = low; int j = mid + 1; int k = low; while (i<=mid && j<=high) { /*每次插入两个数组的头部最小值到目标数组中,保证了目标数组是有序的*/ if (src[i] < src[j]) { dst[k++] = src[i++]; } else { dst[k++] = src[j++]; } } /*将剩下的那部分copy到目标数组中去*/ while (i<=mid) { dst[k++] = src[i++]; } while (j<=high) { dst[k++] = src[j++]; } } int MSort(int* src, int* dst, int low, int high, int max) { assert(src != NULL); assert(dst != NULL); if (low == high) { /*只有一个元素的时候直接给目标数组赋值了。不用在递归啦*/ dst[low] = src[low]; return 0; } else { int mid = (low + high) / 2; int *space = (int*)malloc(sizeof(int)* max); if (space == NULL) { return -1; } //递归调用进行分割直到只有一个元素递归结束 MSort(src, space, low, mid, max); MSort(src, space, mid + 1, high, max); //对递归的数据进行合并 Merge(space, dst, low, mid, high); if (space != NULL) { free(space); } return 0; } return 0; } int MergeSort(int* str,int length) { assert(str != NULL); return MSort(str, str, 0, length - 1, length); }
sortTest.cpp文件
#include <stdio.h> #include <stdlib.h> #include <string.h> #include "Sort.h" int main() { int str1[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str2[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str3[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str4[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str5[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str6[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; int str7[12] = { 12, 4, 53, 3, 2, 34, 6, 35, 1, 132, 324, 12 }; printf("打印排序前的原始结果:\n"); PrintArraryElement(str1, sizeof(str1) / sizeof(*str1)); /*选择排序*/ printf("执行选择排序的结果如下:\n"); SelectSort(str1, sizeof(str1) / sizeof(*str1)); PrintArraryElement(str1, sizeof(str1) / sizeof(*str1)); /*冒泡排序*/ printf("执行冒泡排序的结果如下:\n"); BubbleSort(str2, sizeof(str2) / sizeof(*str2)); PrintArraryElement(str2, sizeof(str2) / sizeof(*str2)); /*插入排序*/ printf("执行插入排序的结果如下:\n"); InsertSort(str3, sizeof(str3) / sizeof(*str3)); PrintArraryElement(str3, sizeof(str3) / sizeof(*str3)); /*希尔排序*/ printf("执行希尔排序的结果如下:\n"); ShellSort(str4, sizeof(str4) / sizeof(*str4)); PrintArraryElement(str4, sizeof(str4) / sizeof(*str4)); /*快速排序1*/ printf("执行快速排序1的结果如下:\n"); QuickSort(str5, sizeof(str5) / sizeof(*str5)); PrintArraryElement(str5, sizeof(str5) / sizeof(*str5)); /*快速排序2*/ printf("执行快速排序(只排序前6个)的结果如下:\n"); QSort(str6, 0,5 ); PrintArraryElement(str6, sizeof(str6) / sizeof(*str6)); /*归并排序*/ printf("执行归并排序的结果如下:\n"); MergeSort(str7, sizeof(str7) / sizeof(*str7)); PrintArraryElement(str7, sizeof(str7) / sizeof(*str7)); system("pause"); return 0; }
测试运行结果如下
posted on 2016-01-12 11:43 LinuxPanda 阅读(543) 评论(0) 编辑 收藏 举报