思路:比较相邻的两个数字,如果前一个数字大,那么就交换两个数字,直到有序。
void bubble_sort(int arr[],size_t len){
size_t i,j;
for(i=0;iarr[j]){ //如果前一个比后一个大
swap(&arr[j-1],&arr[j]); //交换两个数据
hasSwap = true;
}
}
if(!hasSwap){
break;
}
}
}
思路:把一个数字插入一个有序的序列中,使之仍然保持有序,如对于需要我们进行排序的数组,我们可以使它的前i个数字有序,然后再插入i+1个数字,插入到合适的位置使之仍然保持有序,直到所有的数字有序。
void insert_sort(int arr[],int len)
{
int i,j;
for(i=1;i=0&&arr[j]>key;j--)
{ //找到插入的位置
arr[j+1] = arr[j]; //把需要插入的元素后面的元素往后移
}
arr[j+1] = key; //插入该元素
}
}
思路:本质上是插入排序,但是通过半分查找法找到插入的位置,让效率稍微快一点。
void half_insert_sort(int arr[],int len)
{
int i,j;
for(i=1;i=left;j--)
{ //把后面的元素往后移
arr[j+1]=arr[j];
}
arr[j+1] = key; //插入元素
}
}
思路:先取一个正整数d1 思路:通过循环找到最大值所在的位置,然后把最大值和最后一个元素进行交换,通过循环直到所有的数据有序。 思路:选择排序的一种改进,一次循环直接找到最大值和最小值的位置,把最大值和最后一个元素进行交换,最小值和最前一个元素进行交换,所以最外层的循环只需要执行len/2次即可 思路:把数据进行大堆化,然后依次交换堆顶(最大值)和最后一个元素,在使堆顶重新大堆化,最后循环过后数组便有序。 思路:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。 (1)首先设定一个分界值,通过该分界值将数组分成左右两部分 (2)将大于或等于分界值的数据集中到数组右边,小于分界值的数据集中到数组的左边。此时,左边部分中各元素都小于或等于分界值,而右边部分中各元素都大于或等于分界值。 (3)然后,左边和右边的数据可以独立排序。对于左侧的数组数据,又可以取一个分界值,将该部分数据分成左右两部分,同样在左边放置较小值,右边放置较大值。右侧的数组数据也可以做类似处理。 (4)重复上述过程,可以看出,这是一个递归定义。通过递归将左侧部分排好序后,再递归排好右侧部分的顺序。当左、右两个部分各数据排序完成后,整个数组的排序也就完成了。 思路:对于两个有序的子序列,可以把它们合并在一起,变成一个新的完全有序的序列,因此归并排序和快排差不多,都是递归的进行。 思路:这是一种基于比较的算法,我们用一个大数组来存放这些数据,这些数据在这个大数组中的表现形式是以这个大数组的下标存在的,比如57,60,42这三个数字进行排序,那么用一个大数组,这个大数组的arr[57] = 1,arr[60] = 1,arr[42] = 1,然后遍历这个大数组就行了。 思路:工作的原理是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。 代码实现: 基数排序(radix sort)属于“分配式排序”(distribution sort),又称“桶子法”(bucket sort)或bin sort,顾名思义,它是透过键值的部份资讯,将要排序的元素分配至某些“桶”中,藉以达到排序的作用,基数排序法是属于稳定性的排序,其时间复杂度为O (nlog®m),其中r为所采取的基数,而m为堆数,在某些时候,基数排序法的效率高于其它的稳定性排序法。 解法: 1.首先根据个位数的数值,在走访数值时将它们分配至编号0到9的桶子中; 2.接下来将这些桶子中的数值重新串接起来,接着再进行一次分配,这次是根据十位数来分配; 3.接下来将这些桶子中的数值重新串接起来,持续进行以上的动作直至最高位数为止。 代码实现: =========以上内容来自:用C语言完整实现12种排序方法_c语言快速排序解决排序问题-CSDN博客============
void shell_sort(int arr[],int len)
{ //本质上也是一种插入排序,避免了大量数据的移动,在每一组排序过后,每个数据已经到了大致的位置。
int i,j;
int step=0;
for(step = len/2;step>=1;step=step/2)
{ //分组 分为step组,对每组的元素进行插入排序
for(i=step;i
5.选择排序
代码实现:void select_sort(int arr[],size_t len)
{
size_t i,j;
for(i=0;i
6.鸡尾酒排序
代码实现:void cocktail_sort(int arr[],size_t len)
{
size_t i,j;
for(i=0;i
7.堆排序
最大堆调整(Max Heapify):将堆的末端子节点作调整,使得子节点永远小于父节点
创建最大堆(Build Max Heap):将堆中的所有数据重新排序
堆排序(HeapSort):移除位在第一个数据的根节点,并做最大堆调整的递归运算
代码实现:void re_heap(int arr[],size_t index,size_t len)
{
size_t child = 2*index+1; //左节点坐标
int key = arr[index]; //当前节点值
while(child
8.快速排序
过程:
void quick_sort(int arr[],size_t left,size_t right)
{
if(left>=right)
{ //如果只有一个元素,那就是有序的,返回
return;
}
int i = left;
int j = right;
int key = arr[left]; //基准值
while(i
9.归并排序
代码实现:void merge(int arr[],int left,int right)
{
int i,j,k;
int mid = (left+right)/2;
int len = mid-left+1;
int *temp = malloc(sizeof(arr[0])*len);
for(i=0;i
10.计数排序
void count_sort(int arr[],size_t len)
{
int max = arr[0]; //最大值
int min = arr[0]; //最小值
size_t i;
for(i=0;i
11.桶排序
这是一种以消耗大量空间来换取高效率的排序方式,
首先定义桶这个类型:typedef struct Bucket
{
int vect[100]; //其实这里使用链表更好,但是我比较懒,就懒得用链表了
int cnt; //当前桶内存放数据的个数
}Bucket;
void bucket_sort(int arr[],size_t len)
{
int min = arr[0];
int max = arr[0];
size_t i;
for(i=0;i
12.基数排序
还是定义桶的类型:typedef struct Bucket{
int vect[100]; //同样的可以用链表
int cnt;
}Bucket;
void base_sort(int arr[],size_t len)
{
size_t i;
Bucket bucket[10] = {}; //十个桶
int max = arr[0];
for(i=0;i