九种排序具体实现代码

声明:本文内容纯属博主自己查找和归纳的个人所需的知识点,仅作参考,如有错误,博主强烈希望您指出。如果您是某个知识点的原创博主,如有需要,可联系本人加上链接。本文内容会根据博主所需进行更新,希望大家多多关照。

直接插入排序

void InsertSort(int r[])
{
    int n = sizeof(r) / sizeof(r[0]);
    for(int i = 1; i < n; ++i)
    {
        for(int j = i - 1; j >= 0; --j)
        {           
            if(r[j+1] < r[j])
            {
                int s = r[j+1];
                r[j+1] = r[j];
                r[j] = s;
            }
        }
    }
}

折半插入排序

void BinInsertSort(int r[])
{
    int n = sizeof(r) / sizeof(r[0]);
    for(int i = 1; i < n; ++i)
    {
        int s = r[i];
        int low = 0;
        int high = i - 1;
        
        while(low <= high)
        {
            int mid = (low + high) / 2;  //mid位置为要找的数
            if(s < r[mid])
                high = mid - 1;
            else
                low = mid + 1;
        }
        
        for(int j = i - 1; j >= high + 1; --j)  //high+1即mid,执行数的后移,直到mid的数后移
            r[j+1] = r[j];
        
        r[high+1] = s;  //mid位置就存放本身
    }
}

希尔排序

void ShellSort(int r[])
{
    int n = sizeof(r) / sizeof(r[0]);
    int step = n / 2;
    
    while(step >= 1)
    {
        for(int i = step; i < n; ++i)
        {
            for(int j = i - step; j >= 0; j -= step)
            {               
                if(r[j+step] < r[j])
                {
                    int s = r[j+step];
                    r[j+step] = r[j];
                    r[j] = s;
                }
            }
        }
        step /= 2;
    }
}

直接选择排序

void SelectSort(int r[])
{
    int n = sizeof(r) / sizeof(r[0]);
    for(int i = 0; i < n - 1; ++i)
    {
        int samll = i;
        for(int j = i + 1; j < n; ++j)
        {
            if(r[small] > r[j])
                samll = j;
        }
        if(small != i)
        {
            int s = r[i];
            r[i] = r[small];
            r[small] = s;
        }
    }
}

堆排序

void HeapAdjust(int r[]; int i; int j)  //调整堆
{
    int child = 2 * i;
    int s = r[i];  //s临时存放结点数据
    while(child <= j)
    {
        if(child < j && r[child+1] > r[child])  //比较2个子树
            ++child;
        if(s >= r[child])  //结点与大子树比较
            break;
        r[child/2] = r[child];  //如果大子树比结点大,互换
        child = 2 * child;  //继续向子树检索
    }
    r[child/2] = s;  //结点的数为最大的数
}

void HeapSort(int r[])  //建堆
{
    int n = sizeof(r) / sizeof(r[0]);
    for(int i = n / 2 - 1; i >= 0; --i)  //只有n/2-1前的下标才有子树
    {
        HeapAdjust(r, i, n - 1);  //构造大顶堆,结点都比子树大,最后根节点为最大的数
    }
    for(int i = n - 1; i > 0; --i)
    {
        //将当前堆顶元素与当前堆尾元素互换,即将最大的数移到末尾
        int s = r[0];
        r[0] = r[i];
        r[i] = s;
        HeapAdjust(r, 0, i -1);  //将剩下的元素继续调整,最后变成由小到大的顺序
    }
}

冒泡排序

void BubbleSort(int r[])
{
    int n = sizeof(r) / sizeof(r[0]);
    for(int i = 0; i < n - 1; ++i)
    {
        for(int j = 0; j < n - 1 - i; ++j)
        {
            if(r[j] > r[j+1])
            {
                int s = r[j];
                r[j] = r[j+1];
                r[j+1] = s;
            }
        }
    }
}

快速排序

int Partition(int r[], int low, int high)
{
    int pivotkey = r[low];
    int i = low;
    int j = high;
    
    while(i < j)
    {
        while(i < j && r[j] > pivotkey)  //从j往前找,找出第一个比pivotkey小的数
            --j;
        if(i < j)
        {
            r[i] = r[j];
            ++i;
        }
        
        while(i < j && r[i] < pivotkey)  //从i往后找,找出第一个比pivotkey大的数
            ++i;
        if(i < j)
        {
            r[j] = r[i];
            --j;
        }
    }
    r[j] = pivotkey;  //完成最终交换
    return j;  //返回分界点,前面的数都比pivotkey小,后面的数都比pivokey大
}

void QuickSort(int r[], int low, int high) // 传数组、0和长度-1
{
    if(low < high)
    {
        int pivot = Partition(r, low, high);
        QuickSort(r, low, pivot - 1);  //递归,前半部分继续进行快速排序
        QuickSort(r, pivot + 1; high);  //递归,后半部分继续进行快速排序
    }
}

归并排序

void copyArray(int source[], int dest[], int len, int first)
{
    int i;
    int j = first;
    for(i = 0; i < len; ++i)
    {
        dest[j] = source[i];
        ++j;
    }
}

void merge(int a[], int left, int right)
{
    int begin1 = left;
    int mid = (left + right) / 2;
    int begin2 = mid + 1;
    int newArrayLen = right - left + 1;
    int *b = (int*)malloc(newArrayLen * sizeof(int));
    int k = 0;
    
    while(begin1 <= mid && begin2 <= right)  //找出2组中比较后小的那个数按顺序放进b空间
    {
        if(a[begin1] <= a[begin2])
            b[k++] = a[begin1++];
        else
            b[k++] = a[begin2++];
    }
    
    //把剩下的数放进b空间
    while(begin1 <= mid)
        b[k++] = a[begin1++];
    while(begin2 <= right)
        b[k++] = a[begin2++];
    
    copyArray(b, a, newArrayLen, left);  //把b空间的数写回原数组
    free(b);
}

void  MergeSort(int r[], int left, int right)  //传数组、0和长度-1
{
    int i;
    //至少有两个元素才进行排序
    if(left < right)
    {
        i = (left + right) / 2;
        MergeSort(r, left, i);  //前半部分递归
        MergeSort(r, i + 1, right); //后半部分递归        
        merge(r, left, right);  //10个数的比较顺序为[0,1][0,2][3,4][0,4][5,6][5,7][8,9][5,9][0,9]
    }
}

桶排序

void insert(list<int> &bucket, int val)
{
    auto iter = bucket.begin();
    while(iter != bucket.end() && val >= *iter)
        ++iter;
    //insert会在iter之前插入数据,这样可以稳定排序
    bucket.insert(iter, val);
}

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

推荐阅读更多精彩内容