经典排序算法与STL

排序算法

按照是否将元素放入到内存中,排序分为内部排序和外部排序。内部排序适合元素不多的文件,按照元素的排序原则,内部排序分为:

  1. 插入排序
  2. 交换排序
  3. 选择排序
  4. 归并排序

插入排序

基本方法是:寻找一个指定元素在待排序元素中的位置,然后将它插入

直接插入

基本思想:每次将一个待排序的元素按其关键码的大小插入到一个已经排好序的有序序列中,知道全部元素排序好。

void insert_sort(int arr[], int len){
 for (int i=1; i<len; i++) {
    int temp = arr[i] ;
    for (int j = i-1; j>=0; j--) {
      //j+1位置一直是空的,
        if (arr[j]>temp) {
            arr[j+1] = arr[j] ;
        }else {
            arr[j+1] = temp ;
            break ;
        }
    }
    
  }

 }

直接插入的两个特点: 基本有序的序列,直接插入最快。 记录个数很少的无序序列,直接插入也很快

希尔排序

希尔排序是对直接插入排序的一种改进。它的基本思想是:将待排序的元素分成多个子集,分别对这些子集进行直接插入排序,待整个序列都有序时,再对元素进行一次直接插入排序。

void Sort::shell_sort(int arr[] ,int len) {
//增长区间
for (int d = len /2; d>=1; d = d/2)
{   //一次希尔排序
    for (int i = d; i < len; i++)
    {    //和有序的数组的最后一个元素比较
        if (arr[i]<arr[i-d])
        {
            int temp = arr[i] ;
            int j = i-d ;
            while(j>=0&&arr[j]>temp){
                arr[j+d] = arr[j];
                j = j-d;
            }
            arr[j+d] = temp ;
        }
    }
}

}

交换排序

交换排序的基本方法是:在待排序的元素中选择两个元素,将他们的关键码进行比较,进行排序。

冒泡排序

基本思想是:两两比较相邻的元素,如果反序,则交换位置,直到没有反序的元素为止。

void bubble_sort(int arr[], int len){
//从小到大
for (int i = 0; i<len; i++) {
    for (int j = len-1; j>i; j--) {
        if (arr[j]<arr[j-1]) {
            int temp = arr[j] ;
            arr[j] = arr[j-1] ;
            arr[j-1] = temp ;
        }
    }
}    
}

快速排序

快速排序是冒泡排序的改进算法,快速排序元素的比较设移动是序列的两端到中间进行的。它的基本思想是:
在分区中选择一个元素为轴值,将待排序的元素划分为两个分区,左侧元素的关键码都小于或者等于轴值,右侧元素的关键码的值都大于或者等于轴值。重复上述过程,直到整个序列有序。

void quick_sort(int arr[], int left, int right){
//递归结束条件
if (left<right) {
    //一次遍历
    int i = left , j = right , key = arr[left] ;
     while (i<j) {
        while (i<j&&arr[j]>key) 
        j-- ;
        if (i<j)
        arr[i++] =arr[j];
        while (i<j&& arr[i]<key)
        i++ ;
        if (i<j) 
        arr[j--] = arr[i] ; 
     }
    //递归
    arr[i]=key ;
    quick_sort(arr, left, i-1);
    quick_sort(arr, i+1, right); 
}

}

选择排序

选择排序的基本思想是:每趟排序在待排序列表中选择关键码最小的元素,顺序添加到已将排序好的有序序列中去,直到全部记录都排序完成

简单选择排序

简单选择排序的基本思想是:每趟在待排序的序列中找出最小的记录加到序列头部。直到都排序完成。

void select_sort(int arr[], int len){
for (int i= 0; i<len-1; i++) {
        int temp = i ;
    for (int j=i+1; j<len; j++) {
        if(arr[j]<arr[temp]){
            temp = j ;
        }
    }
    if (temp != i) {
        int arrTemp = arr[i] ;
        arr[i] = arr[temp] ;
        arr[temp] =arrTemp ;
    }
}
}

堆排序

堆排序是对简单选择排序的改进。堆排序利用了前一趟比较的结果,减小了比较的次数,从而提高整个排序的效率。

//调整堆
void Sort::heap_adjust(int arr[], int i, int len){
int child = 2 * i + 1;
int temp;
while (child<len) {
 // 得到子结点中键值较大的结点 这里求的是大根堆, 如果是小根堆,下面变为arr[child + 1] < arr[child]
    if (child < len - 1 && arr[child + 1]>arr[child])
        child ++;
    //这里求的是大根堆,如果求小根堆arr[i] > arr[child]
    if (arr[i] <arr[child])
    {
        temp = arr[i];
        arr[i] = arr[child];
        arr[child] = temp;
        i =child ;
        child = 2 * i + 1;
    }
    else
        break;
}   
}

//堆排序
void Sort::heap_sort(int arr[], int len){
int i;
// 调整序列的前半部分元素,调整完之后第一个元素是序列的最大的元素
for (int i = len / 2 - 1; i >= 0; i--)
{
    heap_adjust(arr, i, len);
}

for (i = len - 1; i > 0; i--)
{
    int temp = arr[0];
    arr[0] = arr[i];
    arr[i] = temp;
    // 不断缩小调整heap的范围,每一次调整完毕保证第一个元素是当前序列的最大值
    heap_adjust(arr, 0, i);
}

}

归并排序

归并排序的基本思想方法是:将两个或者两个以上的有序序列归并成一个有序序列

void merge(int arr[], int temp_arr[], int start_index, int mid_index, int end_index){
int i=start_index;
int k=0;
int j=mid_index+1 ;

//直到一个组数被遍历完
while (i<mid_index+1 && j<1+end_index) {
    if (arr[i]>arr[j]) {
        temp_arr[k++] =arr[j++];
    }else{
        temp_arr[k++] =arr[i++];
    }
}
//将没被遍历完的数组添加到数组中去
while (i<1+mid_index) {
    temp_arr[k++]= arr[i++];
}

while (j<1+end_index) {
    temp_arr[k++] =arr[j++];
}

for (i = 0, j = start_index; j <1+end_index; i ++, j ++)
     arr[j] = temp_arr[i];
}

void merge_sort(int arr[], int arr_temp[],int start_index, int end_index){
if (start_index<end_index) {
    int mid = (start_index+end_index)/2 ;
    merge_sort(arr, arr_temp, start_index, mid);
    merge_sort(arr, arr_temp, mid+1, end_index);
    merge(arr, arr_temp, start_index, mid, end_index);
}

排序的比较

排序方法 | 平均复杂度 | 空间复杂度 | 稳定性
------------ | ------------- | ------------
直接插入排序 | O(n2)| O(1) | 稳定
希尔排序 | O(nlog2n) | O(1) |不稳定
冒泡排序 | O(n2)|O(1)|稳定
快速排序 | O(nlog2n) | O(nlog2n)|不稳定
简单选择排序 | O(n2) | O(1)| 稳定
堆排序 | O(nlog2n) | O(1)| 不稳定
归并排序 | O(nlog2n) | O(n)| 稳定

STL中相关排序算法

所有STL sort算法函数的名字列表:

函数名 功能描述
sort 对给定区间所有元素进行排序
stable_sort 对给定区间所有元素进行稳定排序
partial_sort 对给定区间所有元素部分排序
partial_sort_copy 对给定区间复制并排序
nth_element 找出给定区间的某个位置对应的元素
is_sorted 判断一个区间是否已经排好序
partition 使得符合某个条件的元素放在前面
stable_partition 相对稳定的使得符合某个条件的元素放在前面

比较函数:

当你需要按照某种特定方式进行排序时,你需要给sort指定比较函数,否则程序会自动提供给你一个比较函数

vector < int > vect;

sort(vect.begin(), vect.end());//此时相当于调用

sort(vect.begin(), vect.end(), less<int>() );

sort 中的其他比较函数

equal_to 相等
not_equal_to 不相等
less 小于
greater 大于
less_equal 小于等于
greater_equal 大于等于

上述例子中系统自己为sort提供了less仿函数。在STL中还提供了其他仿函数,以下是仿函数列表: 不能直接写入仿 函数的名字,而是要写其重载的()函数: less<int>();
当你的容器中元 素时一些标准类型(int float char)或者string时,你可以直接使用这些函数模板。但如果你时自己定义的类型或者你需要按照其他方式排序,你可以有两种方法来达到效果:一种是自己写比较函数。另一种是重载类型的'<'操作赋。

全排序:

全排序即把所给定范围所有的元素按照大小关系顺序排列。sort采用的是成熟的"快速排序算法"(目前大部分STL版本已经不是采用简单的快速排序,而是结合内插排序算法)。复杂度为nlog2n。stable_sort采用的是"归并排序",分派足够内存时,其算法复杂度为nlog2n, 否则 其复杂度为nlog2n*log2n,其优点是会保持相等元素之间的相对位置在排序前后保持一致。

用于全排序的函 数有:

  1. void sort(RandomAccessIterator first, RandomAccessIterator last);

  2. void sort(RandomAccessIterator first, RandomAccessIterator last,StrictWeakOrdering comp);

  3. void stable_sort(RandomAccessIterator first, RandomAccessIterator last);

  4. void stable_sort(RandomAccessIterator first, RandomAccessIterator last, StrictWeakOrdering comp);

局部排序:

partial_sort采用的堆排序(heapsort),它在任何情况下的复杂度都是nlog2n。

局部排序其实是为了减少不必要的操作而提供的排序方式。

其函数原型为:

  1. void partial_sort(RandomAccessIterator first, RandomAccessIterator middle,RandomAccessIterator last);

  2. void partial_sort(RandomAccessIterator first,RandomAccessIterator middle,RandomAccessIterator last, StrictWeakOrdering comp);

  3. RandomAccessIterator partial_sort_copy(InputIterator first, InputIteratorlast,RandomAccessIteratorresult_first,RandomAccessIterator result_last);

  4. RandomAccessIterator partial_sort_copy(InputIterator first, InputIteratorlast,RandomAccessIteratorresult_first,RandomAccessIterator result_last, Compare comp);

例如:班上有1000个学生,我想知道分数最低的5名是哪些人。
partial_sort(vect.begin(),vect.begin()+5,vect.end(),less<student>());

nth_element 指定元素排序

  1. void nth_element(RandomAccessIterator first, RandomAccessIterator nth, RandomAccessIterator last);

  2. void nth_element(RandomAccessIterator first, RandomAccessIterator nth,RandomAccessIterator last,StrictWeakOrdering comp);

例如:班上有1000个学生,我想知道分数排在倒数第4名的学生。
nth_element(vect.begin(), vect.begin()+3, vect.end(),less<student>());

partition 和stable_partition

partition就是把一个区间中的元素按照某个条件分成两类,并没有排序。

其函数原型为:

  1. ForwardIterator partition(ForwardIterator first, ForwardIterator last, Predicate pred)

  2. ForwardIterator stable_partition(ForwardIterator first, ForwardIterator last, Predicate pred);

例如:班上10个学生,计算所有没有及格(低于60分)的学生:

 student exam("pass", 60);

 stable_partition(vect.begin(), vect.end(), bind2nd(less<student>(), exam));

效率由高到低(耗时由小变大)

partion | |
-----|
stable_partition |
nth_element|
partial_sort|
sort|
stable_sort|

Effective STL对如何选择排序函数总结的很好:

  1. 若需对vector, string, deque, 或array容器进行全排序,你可选择sort或stable_sort;

  2. 若只需对vector, string, deque, 或array容器中取得top n的元素,部分排序partial_sort是首选.

  3. 若对于vector, string, deque,或array容器,你需要找到第n个位置的元素或者你需要得到top n且不关系top n中的内部顺序,nth_element是最理想的;

  4. 若你需要从标准序列容器或者array中把满足某个条件 或者不满足某个条件的元素分开,你最好使用partition或stable_partition;

  5. 若使用的list容器,你可以直接使用partition和stable_partition算法,你可以使用list::sort代替sort和stable_sort排序。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容

  • STL算法部分主要由头文件,,组成。要使用STL中的算法函数必须包含头文件,对于数值算法须包含,中则定义了一些模板...
    eb51589b1211阅读 597评论 0 1
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 1,235评论 0 2
  • 作者:郭耀华cnblogs.com/guoyaohua/p/8600214.html 最近几天在研究排序算法,看了...
    xiaotian是个混子阅读 1,232评论 0 8
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    zwb_jianshu阅读 1,099评论 0 0
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    printf200阅读 759评论 0 0