常见排序主要有以下四种:

1.交换排序

2.选择排序

3.插入排序

4.归并排序

(以下代码基本都有输出每步排序结果)

一.交换排序

交换排序主要是冒泡排序和快排

1.冒泡排序

基本方法:

设待排序对象序列中的对象 个数为 n。

最多作 n-1 趟排序。在第 i 趟中顺次两两 比较r[j-1].Key和r[j].Key,j = i, i+1, ……, n-i-1。

如果发生逆序,则交换r[j-1]和r[j]。

流程:
(1)对数组中各个数字,一次比较相邻两个
(2)如果前面大于后面,就交换这两个数据
(3)再用同样的方法继续排,直到外层循环排完
 
 
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void BubbleSort(int a[],int len)
{
int t;
for (int i = ; i < len-; i++)
{
for (int j = len-; j>i; j--)
{
if (a[j]<a[j-]) //比较相邻两个,前大于后,交换
{
t=a[j-];
a[j-]=a[j];
a[j]=t;
}
}
cout<<"Sort results in"<<i+<<" step :"; //输出每一步排序结果
for (int k = ; k <len ; k++) cout<<a[k]<<" ";
cout<<endl;
}
}int main()
{
int array1[];
srand(time(NULL));
cout<<"Array1 before sorting:"<<endl;
for (int i = ; i < ; i++) //利用随机数作为数组输入
{
array1[i]=rand()/;
cout<<array1[i]<<" ";
}
cout<<endl;
BubbleSort(array1,);
cout<<"Array1 after sorting:"<<endl;
for (int i = ; i < ; i++) cout<<array1[i]<<" ";
cout<<endl;return ;
}

2.快速排序

基本思想:

任取待排序对象序列中的某个对象 (例如取第一个对象) 作为枢轴(pivot),

按照该对象的关键字大小,将整个对象序列划分为左右两个子序列:

– 左侧子序列中所有对象的关键字都小于或等于枢轴对象的 关键字

– 右侧子序列中所有对象的关键字都大于枢轴对象的关键字

枢轴对象则排在这两个子序列中间(这也是该对象最终应安放 的位置)。

流程:
(1)首先设定一个分界值,通过分界值将数组分成左右两部分。
(2)将大于或等于分界值的数据集中到数组右边,小于分界值的数据集中到数组的左边。
(3)左右两个部分重复上述排序。
(详细解释见代码)
#include<iostream>
#include<ctime>
#include<cstdlib>
using namespace std;
void QuickSort(int *a,int left,int right)
{
int rt,lt,t,base;
rt=right; //rt为基准分组后的左半部分上边界
lt=left; //lt为基准分组后的右半部分的下边界
base=a[left];
/*这里定义分界值,这个分界值定在哪里都是可以的,定在中间更为直观,若定在其他地方,手动排一下就知道,边界值会换到中间去,和定在中间一个样*/
while (lt < rt)
{
while(a[lt] < base) lt++; //从左边开始寻找大于分界值的值
while(a[rt] > base) rt--; //从右边开始寻找小于边界值的值
if(lt < rt)
{ /*lt小于rt,就交换这两个的值,lt与rt必不会在边界值同一侧,
手动按照算法排一下就知道,lt或rt到了边界值的时候就会停下来,交换的时候就会把边界值换到中间去了*/
t = a[lt] ;
a[lt] = a[rt] ;
a[rt] = t ;
lt++;
rt--;
}
}
while(a[lt]<base) lt++; //因为右半部分可能还到不了下边界,所以需要继续递增
while(a[rt]>base) rt--;//因为左半部分可能还到不了上界,所以需要继续递减
if(lt == rt) lt++; //此处为了避免两个边界模糊不清
if( left < rt ) QuickSort(a,left,rt); //递归对左半部分快排
if( right > lt) QuickSort(a,lt,right); //递归对右半部分快排 }
int main()
{
srand(time(NULL));
int n;
cout<<"Please cin the size of array:"<<endl;//输入数组的大小
cin>>n;
int a[n];
cout<<"Array before sorting is:"<<endl;
for (int i = ; i < n; i++)
{
a[i]=rand()/; //随机数作为数组输入
cout<<a[i]<<" ";
}
cout<<endl;
QuickSort(a,,n-);
cout<<"Array after sorting is:"<<endl;
for (int i = ; i < n; i++) cout<<a[i]<<" ";
cout<<endl;
return ;
}

二.选择排序

常见的选择排序有简单的选择排序和堆排序

1.简单的选择排序

流程:
(1)先从原始数组选择一个最小数据和第一个位置交换
(2)剩下的n-1个数据选择最小的和第二个位置交换
(3)不断重复直到最后执行完成
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void SelectionSort(int a[],int len)
{
int k,t;
for (int i = ; i < len-; i++)
{
k=i;
for (int j = i+; j < len; j++) //选择第i+1个小的数据
{
if (a[j]<a[k]) k=j;
}
if (k!=i) //选好后就可以交换了
{
t=a[k];
a[k]=a[i];
a[i]=t;
}
cout<<"Sort Result in"<<i+<<"step:"<<endl;
for (int k = ; k < len; k++) cout<<a[k]<<" ";
cout<<endl; } }
int main()
{
int array[];
srand(time(NULL));
cout<<"Array before sorting:"<<endl;
for (int i = ; i < ; i++) //随机数作为数组输入
{
array[i]=rand()/;
cout<<array[i]<<" ";
}
cout<<endl;
SelectionSort(array,);
cout<<"Array after sorting:"<<endl;
for (int i = ; i < ; i++) cout<<array[i]<<" ";
cout<<endl;
return ;
}

2.堆排序

堆排序关键是构造堆结构。这是一个完全二叉树结构。在这个树中每个节点对应原始数据一个记录。每个节点满足以下条件:
(1)若按照从小到大排序,要求非叶节点的数据大于等于其左右子节点的数据
(2)若按照从大到小排序,要求非叶节点的数据要小于等于其左右子节点的数据
可以看出,这只规定父节点和子节点的数据之间必须满足的大小关系。
完整的堆排序需要反复经过两个步骤:
(1)构造堆结构,
(2)堆排序输出。
下面介绍如何构造堆结构:
由完全二叉树的下层向上层逐层对父子节点的数据进行比较,使父节点的数据大于等于子节点的数据。(最小堆)
这里需要用到筛选运算不断调整,直到节点满足堆结构为止:
(1)先将原始数据排成一个二叉树
(2)对最后一个非叶节点进行筛运算。
*筛运算:比较左右子树,最大值放右子树,比较右子树与非叶节点,最大值放非叶节点。(比较过程中并非直接等,而是值互换)
(3)对倒数第二个非叶节点进行筛运算,
(4)一直重复,直到根节点进行筛运算为止。(若根节点的值被互换了,那么就要对互换的节点进行筛运算,一直下去)这时就形成了堆结构。
形成了堆结构,接下来就是堆排序输出(现在是按照从小到大的方式排序):
(1)输出根节点,把最后一个结点的数放到根节点
(2)重新构造堆结构
(3)重复(1)(2)步,直到全部输出。
 
(这里并不构建树,直接对数组操作,关于树如果不理解的可以自己依据完全二叉树画法,首元素为树根,依次往下得出树结构,详细见代码及代码注释)
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void HeapSort(int a[],int n)
{
int t;
for (int i = n/-; i >=; i--) //第n/2-1个节点为止,都是非叶子节点
{
while (*i+<n) //节点2*i+1是i的子树
{
int j=*i+;
if (j+<n) //判定是否为一个子树
{ //不是一个子树
if(a[j]<a[j+]) j++; //判定哪个子树大,较大的子树与父节点比较
} if (a[i]<a[j]) //与父节点比较
{
t=a[i]; //交换
a[i]=a[j];
a[j]=t;
i=j; //因为与父节点的值交换了,所以需要对交换的节点重新进行筛选运算
}
else break;
} }
cout<<"The original heap structure is:"<<endl; //输出原始堆结构
for(int i=;i<n;i++) cout<<a[i]<<" ";
cout<<endl;
int i;
for (i = n-; i >=; i--) //开始排序
{
t=a[]; //交换根节点的值到数组最后
a[]=a[i];
a[i]=t;
int k=;
while (*k+<i) //开始重新构造堆结构,因为从i开始,所以到i截止
{
int j=*k+; //接下来与构造堆结构同
if (j+<i)
{
if(a[j]<a[j+]) j++;
}
if (a[k]<a[j])
{
t=a[k];
a[k]=a[j];
a[j]=t;
k=j;
}
else break;
}
cout<<"Array after "<<n-i<<"step:"; //输出每一步排序
for (int c = ; c < n; c++) cout<<a[c]<<" ";
cout<<endl; } }
int main()
{
srand(time(NULL)); //随机数作为数组输入
int n;
cout<<"Please cin the size of array:"<<endl;
cin>>n;
int a[n];
cout<<"Array before sorting:"<<endl;
for (int i = ; i < n; i++)
{
a[i]=rand()/;
cout<<a[i]<<" ";
}
cout<<endl;
HeapSort(a,n);
cout<<"Array after sorting:"<<endl;
for(int i=;i<n;i++) cout<<a[i]<<" ";
cout<<endl;
return ;
}

三.插入排序

插入排序:通过对未排序的数据逐个插入合适的位置而完成排序工作

常见的插入排序有简单的插入排序和Shell排序

1.直接的插入排序

直接插入排序的基本思想是:

当插入第i (i ≥ 1) 个对象时,前面 的v[0], v[1], …, v[i-1]已经排好序。

这时,用v[i]的关键字与v[i1], v[i-2], …的关键字顺序进行比较,

找到插入位置即将v[i]插 入,原来位置上之后的所有对象依次向后顺移。

流程:
(1)先对数组前两个数据进行从小到大排序
(2)将第三个数据与前两个数据比较,将第三个数据插入合适的位置
(3)将第四个数据插入已排序好的前三个数据中
(4)不断重复,直到把最后一个数据插入合适的位置
(详细见代码及注释)
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void InsertionSort(int a[],int len)
{
int t;
for (int i = ; i < len; i++)
{
t=a[i]; //对第i个数进行标记(从零开始算,下同)
int j=i-; //从第i-1个数倒着回去找位置(第i-1个数前面的都从小到大排好了)
while (j>=&&t<a[j]) //找位置,小于第j个数继续找,而比较过的数就往后退留出空位置
{//由于我们标记了第i个数,所以我们比较过的数往后退的时候可以留出一个位置,因为第i个数位置空出来了
a[j+]=a[j];
j--;
}
a[j+]=t;
cout<<"Sort result after"<<i+<<"step:";
for(int k=;k<len;k++) cout<<a[k]<<" ";
cout<<endl;
}
}
int main()
{
int a[];
srand(time(NULL));
cout<<"Array before sorting:"<<endl; //随机数作为数组输入
for (int i = ; i < ; i++)
{
a[i]=rand()/;
cout<<a[i]<<" ";
}
cout<<endl;
InsertionSort(a,);
cout<<"Array after sort:"<<endl;
for (int i = ; i < ; i++) cout<<a[i]<<" ";
cout<<endl;
return ; }

2.Shell排序 (处理数据量比较大的时候的插入排序)

基本思想:

先将整个待排对象序列按照一定间隔分 割成为若干子序列,

分别进行直接插入排序,然后缩小间隔, 对整个对象序列重复以上的划分子序列和分别排序工作,

直到 最后间隔为1,此时整个对象序列已 “基本有序”,进行最后 一次直接插入排序。

流程:
(1)将n个元素数组分成n/2个数字序列,第一个数据和第n/2个数据为一对,等等,以此类推
         比如说 8 6 5 4 2 1 9 7
         就分为数字序列 8 6 5 4 以及 2 1 9 7 
         那么8 和 2就是一个数对
(2)一次循环使每一个数对排列好顺序(假设从小到大)
         依据上述,排序好后就是 2 1 5 4 和 8 6 9 7(2小于8,交换,1小于6,交换,9大于5,7大于4所以不用交换)
(3)变成n/4个数对,再次排序。
         (数字序列就成为 2 1 ,5 4,8 6,9 7,这个时候有三次比较,设四个序列分别为a,b,c,d,那么此时b与a比较,然后c与b比较,然后d与c比较,后面以此类推 )
(4)不断重复上述过程,随着序列减少直至最后变为1个(此时就成为插入排序了),完成排序。
(详细见代码及注释)
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void ShellSort(int *a,int len)
{
int x,t,j;
for (int r = len/; r >= ; r/=) //外层循环分序列
{
for (int i = r; i < len; i++)
{//内层循环设置一定间距r,分别比较对应元素,当r=1时,这个时候就为插入排序,数组量大时这时效率比较高。
t=a[i];
j=i-r;
while (j>=&&t<a[j])
{
a[j+r]=a[j];
j-=r;
}
a[j+r]=t;
x++;
cout<<"Sort result after "<<x<<" step "; //输出每一步排序结果
for (int k = ; k < len; k++) cout<<a[k]<<" ";
cout<<endl;
} } }
int main()
{
srand(time(NULL));
int n;
cout<<"Please cin the size of array:"<<endl; //输入数组的大小
cin>>n;
int a[n];
cout<<"Array before sorting:"<<endl;
for (int i = ; i < n; i++) //利用随机数作为数组输入
{
a[i]=rand()/;
cout<<a[i]<<" ";
}
cout<<endl;
ShellSort(a,n);
cout<<"Array after sorting:"<<endl; //输出排序后的数组
for (int i = ; i < n; i++) cout<<a[i]<<" ";
cout<<endl;
return ; }
四.归并排序
一个待排序的原始数据序列进行排序归并排序的基本思路:
(1)将n个待排序数据看成n个有序子表,这时候无所谓有序无序。
(2)将他们依次两两合并,得到长度为2的有序子表
(3)再两两合并,得到长度为4的有序子表。不能合并的留到下次一起合并
(4)重复上述步骤,直到表长度为n
eg.(1)待排序数据24 30 29 3 2 21 3 8 ,这里就是8个有序子表,无所谓有序无序
     (2)两两合并,对子表排序,得到 24 30 ,3  29,2  21,3  8四个长度为2的有序子表
     (3)再两两合并,得到长度为4的有序子表 3 24 29 30,2  3  8  21
     (4)再合并得到 2 3 3 8 21 24 29 30
详细见代码及注释(非递归实现)
#include<iostream>
#include<cstdlib>
#include<ctime>
using namespace std;
void MergeOne(int *a,int *p,int n,int len)
{
int i,j,k,begin,edge; //i,j分别代表合并前左子表和右子表的初始下标,k表示要赋值的数组的下标
begin=; //begin代表合并的两组子表的初下标
while(begin+len<n)
{
edge=begin+*len-; //edge代表合并后的两组子表的末下标
if(edge>=n) edge=n-; //因为有可能并不能全部一次性合并,有奇数,这时就会剩下未合并的
i=begin;
k=begin;
j=begin+len;
while (i<begin+len&&j<=edge) //当子表下标都在合并范围内的时候
{
if(a[i]<a[j]) p[k++]=a[i++]; //哪个小,哪个就赋值到数组
else p[k++]=a[j++];
}
while(i<begin+len) p[k++]=a[i++];//如果左边子表剩下了,就把剩下的都放进数组
while(j<=edge) p[k++]=a[j++]; //如果右子表剩下了,就把剩下的放进数组
begin=edge+; //继续合并下两组子表
}
if(begin<n) for(;begin<n;begin++) p[begin]=a[begin]; //未合并的,单个的,先放进数组
}
void MergeSort(int *a,int n)
{
int *p=new int [n]; //合并的数组
int len=; //子表长度
int f=; //标记,以a为初始数组还是以p为初始数组
int count=; //记录合并步骤
while (len<n) //合并后子表长度小于n时,可以继续合并
{
if(f==) MergeOne(p,a,n,len); //以p为初始数组
else MergeOne(a,p,n,len); //以a为初始数组
f=-f; //切换
len*=; //合并一次,子表长度double
count++; //合并次数递增
cout<<count<<" step:";
//输出每步结果
if(f) for (int i = ; i < n; i++) cout<<p[i]<<" ";
else for (int i = ; i < n; i++) cout<<a[i]<<" ";
cout<<endl; }
if (f)
{//判断最后一次是否以a为初始数组,如果是的话,就要把p的值全部赋值给a
for(int i=;i<n;i++) a[i]=p[i];
}
delete []p;
}
int main()
{
srand(time(NULL));
cout<<"Please cin the size of array:"<<endl;
int n;
cin>>n;
int a[n];
cout<<"Array before sorting:"<<endl;
for (int i = ; i < n; i++)
{
a[i]=rand()/;
cout<<a[i]<<" ";
}
cout<<endl;
MergeSort(a,n);
cout<<"Array after sorting:"<<endl;
for(int i=;i<n;i++) cout<<a[i]<<" ";
cout<<endl;
return ;
}

归并算法递归实现如下,详情见代码及注释(需要手动输入数组大小及数组的值)

递归主要思想是分治:

思路:(1)先将数组划分为一个个块,直到不能再划分为止(分)

(2)对划分的数组进行排序

需要:(1)一个临时变量数组p,存储每次归并后的值,然后赋给a(这个时候p处理归并数据下标的范围和a的下标范围一致)

(2)一个函数,用来分(划分下标范围)

(3)一个函数,用来治(将两组数据归并,归并后即为排序)

详细请见代码及注释

 #include<bits/stdc++.h>
using namespace std;
void Mergesort(int a[],int begin,int mid,int end,int n,int p[]) //归并排序
{
int i=begin,j=mid+,k=i;
//i是左分组起始下标,mid是分组边界也是i的上界,j是右分组的左边界,j上界为end,k为临时储存数组p的下标
while (i<=mid&&j<=end) //开始对两个分组进行归并
{
if (a[i]<a[j]) p[k++]=a[i++]; //分组中小的放先放进p,然后递增下标
else p[k++]=a[j++];
} //循环结束时,左右两分组必有一个没有全部并到临时储存数组p
while(i<=mid) p[k++]=a[i++]; //这时我们需要检查,并把没有并的放到p里面
while(j<=end) p[k++]=a[j++];
for (int i = ; i < (end-begin+); i++) a[i+begin]=p[i+begin]; //把归并好的放到a里
for(int i=;i<n;i++) cout<<a[i]<<" "; //输出每一步的归并结果
cout<<endl;
} void Divide_Conquer(int a[],int begin,int end,int n,int p[]) //递归分治
{
if (begin<end)
{
int mid=begin+(end-begin)/; //定一个分治的边界
Divide_Conquer(a,begin,mid,n,p); //左分
Divide_Conquer(a,mid+,end,n,p); //右分
Mergesort(a,begin,mid,end,n,p); //治,对分的数据进行排序
} } int main()
{
int n;
cin>>n; //输入数组大小
int a[n],p[n]; //a[n]为数组,p[n]为临时数组
for(int i=;i<n;i++) cin>>a[i]; //输入数组数值
Divide_Conquer(a,,n-,n,p); //开始分治,归并
//cout<<"Array after sort:"<<endl;
//for(int i=0;i<n;i++) cout<<a[i]<<" "; //输出归并的值
//cout<<endl;
return ;
}

以上是常见的排序,望大家轻喷。(狗头保命)

常见排序汇总C&C++的更多相关文章

  1. 常见排序算法总结分析之选择排序与归并排序-C#实现

    本篇文章对选择排序中的简单选择排序与堆排序,以及常用的归并排序做一个总结分析. 常见排序算法总结分析之交换排序与插入排序-C#实现是排序算法总结系列的首篇文章,包含了一些概念的介绍以及交换排序(冒泡与 ...

  2. 常见排序算法(附java代码)

    常见排序算法与java实现 一.选择排序(SelectSort) 基本原理:对于给定的一组记录,经过第一轮比较后得到最小的记录,然后将该记录与第一个记录的位置进行交换:接着对不包括第一个记录以外的其他 ...

  3. JS常见排序算法

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  4. JavaScript版几种常见排序算法

    今天发现一篇文章讲“JavaScript版几种常见排序算法”,看着不错,推荐一下原文:http://www.w3cfuns.com/blog-5456021-5404137.html 算法描述: * ...

  5. 李洪强iOS开发之OC常见错误汇总

    // //  main.m //  16 - 常见错误汇总 // //  Created by vic fan on 16/7/13. //  Copyright © 2016年 李洪强. All r ...

  6. 常见排序算法(JS版)

    常见排序算法(JS版)包括: 内置排序,冒泡排序,选择排序,插入排序,希尔排序,快速排序(递归 & 堆栈),归并排序,堆排序,以及分析每种排序算法的执行时间. index.html <! ...

  7. ios 排序汇总

    ios 排序汇总  IOS几种简单有效的数组排序方法 //第一种,利用数组的sortedArrayUsingComparator调用 NSComparator ,obj1和obj2指的数组中的对象 N ...

  8. 常见排序算法-Python实现

    常见排序算法-Python实现 python 排序 算法 1.二分法     python    32行 right = length-  :  ]   ):  test_list = [,,,,,, ...

  9. (转)JAVA排序汇总

    JAVA排序汇总 package com.softeem.jbs.lesson4; import java.util.Random; /** * 排序测试类 * * 排序算法的分类如下: * 1.插入 ...

随机推荐

  1. C语言作业|08

    问题 答案 这个作业的属于那个课程 C语言程序设计II 这个作业要求在哪里 https://edu.cnblogs.com/campus/zswxy/CST2019-2/homework/9977 我 ...

  2. php使用cUrl方法 get、post请求

    php使用curl方法,请确保已经开启curl扩展.传送门:http://www.cnblogs.com/wgq123/p/7450667.html /**Curl请求get方法 *@$url Str ...

  3. windows版的mysql主从复制环境搭建

    背景 最近在学习用Spring Aop来实现数据库读写分离的功能. 在编写代码之前,首先是要部署好mysql的环境,因为要实现读写分离,所以至少需要部署两个mysql实例,一主一从,并且主从实例之间能 ...

  4. Orleans 3.0 为我们带来了什么

    原文:https://devblogs.microsoft.com/dotnet/orleans-3-0/ 作者:Reuben Bond,Orleans首席软件开发工程师 翻译:艾心 这是一篇来自Or ...

  5. Spring Boot2 系列教程(二十七)Nginx 极简扫盲入门

    上篇文章和大家聊了 Spring Session 实现 Session 共享的问题,有的小伙伴看了后表示对 Nginx 还是很懵,因此有了这篇文章,算是一个 Nginx 扫盲入门吧! 基本介绍 Ngi ...

  6. day 36 html的补充

    参考博客:https://www.cnblogs.com/majj/p/9062540.html 内容回顾: 0.浏览器 1.标签 - 行内标签 a span i em strong b.label ...

  7. 【JavaEE】之MyBatis输出映射

    MyBatis中的输出映射有两种:resultType和resultMap. 1.resultType 使用resultType进行结果映射时,只有当查询结果中有至少一列的名称和resultType指 ...

  8. window.open()打开新窗口 及参数

    在jsp页面中需要使用到弹出窗口,想到js的window对象有一个open方法可以弹出窗口,于是对open方法进行记录. 首先是open方法的语法及定义: 定义: open() 方法用于打开一个新的浏 ...

  9. docker下安装测试环境estuntest

    1.基础知识: docker pull centos   //从云上下载centos系统到本地服务器 docker images   //查看镜像docker rmi 镜像id   //删除镜像 do ...

  10. Jpa支持LocalDateTime类型持久化

    package com.boldseas.porscheshop.common.config; import javax.persistence.AttributeConverter; import ...