0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

C++基础语法十大排序算法后五个分享

C语言编程学习基地 来源:C语言编程学习基地 作者:C语言编程学习基地 2021-10-08 15:06 次阅读

本期是C++基础语法分享的第十六节,今天给大家来梳理一下十大排序算法后五个!

a9519302-2426-11ec-82a8-dac502259ad0.png

归并排序

归并排序:把数据分为两段,从两段中逐个选最小的元素移入新数据段的末尾。可从上到下或从下到上进行。

/***************** 迭代版*****************

///整數或浮點數皆可使用,若要使用物件(class)時必須設定“小於”(《)的運算子功能template《typename T》void merge_sort(T arr[], int len) { T* a = arr;

T* b = new T[len]; for (int seg = 1;

seg 《 len; seg += seg) { for (int start = 0; start 《 len; start += seg + seg) { int low = start, mid = min(start + seg, len), high = min(start + seg + seg, len);

int k = low; int start1 = low, end1 = mid; int start2 = mid, end2 = high; while (start1 《 end1 && start2 《 end2) b[k++] = a[start1] 《 a[start2] ? a[start1++] : a[start2++]; while (start1 《 end1) b[k++] = a[start1++];

while (start2 《 end2) b[k++] = a[start2++]; } T* temp = a;

a = b; b = temp; } if (a != arr) { for (int i = 0; i 《 len; i++) b[i] = a[i]; b = a;

} delete[] b;}

/***************** 递归版*****************/template《typename T》void merge_sort_recursive(T arr[], T reg[], int start, int end) { if (start 》= end) return; int len = end - start, mid = (len 》》 1) + start; int start1 = start, end1 = mid;

int start2 = mid + 1, end2 = end; merge_sort_recursive(arr, reg, start1, end1); merge_sort_recursive(arr, reg, start2, end2);

int k = start; while (start1 《= end1 && start2 《= end2) reg[k++] = arr[start1] 《 arr[start2] ? arr[start1++] : arr[start2++];

while (start1 《= end1) reg[k++] = arr[start1++]; while (start2 《= end2) reg[k++] = arr[start2++];

for (k = start; k 《= end; k++) arr[k] = reg[k];}//整數或浮點數皆可使用,若要使用物件(class)時必須設定“小於”(《)的運算子功能template《typename T》 void merge_sort(T arr[], const int len) { T *reg = new T[len]; merge_sort_recursive(arr, reg, 0, len - 1); delete[] reg;}

希尔排序

希尔排序:每一轮按照事先决定的间隔进行插入排序,间隔会依次缩小,最后一次一定要是1。

template《typename T》void shell_sort(T array[], int length) { int h = 1; while (h 《 length / 3) { h = 3 * h + 1; } while (h 》= 1) { for (int i = h; i 《 length; i++) { for (int j = i; j 》= h && array[j] 《 array[j - h]; j -= h) { std::swap(array[j], array[j - h]); } } h = h / 3; }}

计数排序

计数排序:统计小于等于该元素值的元素的个数i,于是该元素就放在目标数组的索引i位(i≥0)。

计数排序基于一个假设,待排序数列的所有数均为整数,且出现在(0,k)的区间之内。

如果 k(待排数组的最大值) 过大则会引起较大的空间复杂度,一般是用来排序 0 到 100 之间的数字的最好的算法,但是它不适合按字母顺序排序人名。

计数排序不是比较排序,排序的速度快于任何比较排序算法。

时间复杂度为 O(n+k),空间复杂度为 O(n+k)

算法的步骤如下:

1. 找出待排序的数组中最大和最小的元素

2. 统计数组中每个值为 i 的元素出现的次数,存入数组 C 的第 i 项

3. 对所有的计数累加(从 C 中的第一个元素开始,每一项和前一项相加)

4. 反向填充目标数组:将每个元素 i 放在新数组的第 C[i] 项,每放一个元素就将 C[i] 减去 1

#include 《iostream》#include 《vector》#include 《algorithm》

using namespace std;

// 计数排序void CountSort(vector《int》& vecRaw, vector《int》& vecObj){ // 确保待排序容器非空 if (vecRaw.size() == 0) return;

// 使用 vecRaw 的最大值 + 1 作为计数容器 countVec 的大小 int vecCountLength = (*max_element(begin(vecRaw), end(vecRaw))) + 1; vector《int》 vecCount(vecCountLength, 0);

// 统计每个键值出现的次数 for (int i = 0; i 《 vecRaw.size(); i++) vecCount[vecRaw[i]]++; // 后面的键值出现的位置为前面所有键值出现的次数之和 for (int i = 1; i 《 vecCountLength; i++) vecCount[i] += vecCount[i - 1];

// 将键值放到目标位置 for (int i = vecRaw.size(); i 》 0; i--) // 此处逆序是为了保持相同键值的稳定性 vecObj[--vecCount[vecRaw[i - 1]]] = vecRaw[i - 1];}

int main(){ vector《int》 vecRaw = { 0,5,7,9,6,3,4,5,2,8,6,9,2,1 }; vector《int》 vecObj(vecRaw.size(), 0);

CountSort(vecRaw, vecObj);

for (int i = 0; i 《 vecObj.size(); ++i) cout 《《 vecObj[i] 《《 “ ”; cout 《《 endl;

return 0;}

桶排序

桶排序:将值为i的元素放入i号桶,最后依次把桶里的元素倒出来。

桶排序序思路:

1. 设置一个定量的数组当作空桶子。

2. 寻访序列,并且把项目一个一个放到对应的桶子去。

3. 对每个不是空的桶子进行排序。

4. 从不是空的桶子里把项目再放回原来的序列中。

假设数据分布在[0,100)之间,每个桶内部用链表表示,在数据入桶的同时插入排序,然后把各个桶中的数据合并。

const int BUCKET_NUM = 10;

struct ListNode{ explicit ListNode(int i=0):mData(i),mNext(NULL){} ListNode* mNext; int mData;};

ListNode* insert(ListNode* head,int val){ ListNode dummyNode; ListNode *newNode = new ListNode(val); ListNode *pre,*curr; dummyNode.mNext = head; pre = &dummyNode; curr = head; while(NULL!=curr && curr-》mData《=val){ pre = curr; curr = curr-》mNext; } newNode-》mNext = curr; pre-》mNext = newNode; return dummyNode.mNext;}

ListNode* Merge(ListNode *head1,ListNode *head2){ ListNode dummyNode; ListNode *dummy = &dummyNode; while(NULL!=head1 && NULL!=head2){ if(head1-》mData 《= head2-》mData){ dummy-》mNext = head1; head1 = head1-》mNext; }else{ dummy-》mNext = head2; head2 = head2-》mNext; } dummy = dummy-》mNext; } if(NULL!=head1) dummy-》mNext = head1; if(NULL!=head2) dummy-》mNext = head2; return dummyNode.mNext;}

void BucketSort(int n,int arr[]){ vector《ListNode*》 buckets(BUCKET_NUM,(ListNode*)(0)); for(int i=0;i《n;++i){ int index = arr[i]/BUCKET_NUM; ListNode *head = buckets.at(index); buckets.at(index) = insert(head,arr[i]); } ListNode *head = buckets.at(0); for(int i=1;i《BUCKET_NUM;++i){ head = Merge(head,buckets.at(i)); } for(int i=0;i《n;++i){ arr[i] = head-》mData; head = head-》mNext; }}

基数排序

基数排序:一种多关键字的排序算法,可用桶排序实现。

int maxbit(int data[], int n) //辅助函数,求数据的最大位数{ int maxData = data[0]; ///《 最大数 /// 先求出最大数,再求其位数,这样有原先依次每个数判断其位数,稍微优化点。 for (int i = 1; i 《 n; ++i) { if (maxData 《 data[i]) maxData = data[i];

} int d = 1;

int p = 10; while (maxData 》= p) { //p *= 10; // Maybe overflow maxData /= 10; ++d; } return d;/* int d = 1; //保存最大的位数 int p = 10; for(int i = 0; i 《 n; ++i) { while(data[i] 》= p) { p *= 10; ++d;

} } return d;*/}void radixsort(int data[], int n) //基数排序{ int d = maxbit(data, n);

int *tmp = new int[n]; int *count = new int[10];

//计数器 int i, j, k; int radix = 1; for(i = 1; i 《= d; i++) //进行d次排序 { for(j = 0; j 《 10; j++) count[j] = 0; //每次分配前清空计数器 for(j = 0; j 《 n; j++) { k = (data[j] / radix) % 10; //统计每个桶中的记录数 count[k]++; } for(j = 1; j 《 10; j++) count[j] = count[j - 1] + count[j];

//将tmp中的位置依次分配给每个桶 for(j = n - 1; j 》= 0; j--) //将所有桶中记录依次收集到tmp中 { k = (data[j] / radix) % 10; tmp[count[k] - 1] = data[j];

count[k]--; } for(j = 0; j 《 n; j++) //将临时数组的内容复制到data中 data[j] = tmp[j]; radix = radix * 10;

} delete []tmp; delete []count;}

今天的分享就到这里了,大家要好好学C++哟~

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编程
    +关注

    关注

    88

    文章

    3438

    浏览量

    92321
  • C++
    C++
    +关注

    关注

    21

    文章

    2066

    浏览量

    72880

原文标题:C++基础语法梳理:算法丨十大排序算法(二)

文章出处:【微信号:cyuyanxuexi,微信公众号:C语言编程学习基地】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    十大排序算法总结

    排序算法是最经典的算法知识。因为其实现代码短,应该广,在面试中经常会问到排序算法及其相关的问题。一般在面试中最常考的是快速
    的头像 发表于 12-20 10:39 710次阅读

    排序算法有哪些

    1. 归并排序(递归版) 归并排序(MERGE-SORT)是利用归并的思想实现的排序方法,该算法采用经典的分治策略,即分为两步:分与治。 分:先递归分解数组成子数组 治:将分阶段得到的
    的头像 发表于 10-11 15:49 322次阅读
    <b class='flag-5'>排序</b><b class='flag-5'>算法</b>有哪些

    排序算法之选择排序

    选择排序: (Selection sort)是一种简单直观的排序算法,也是一种不稳定的排序方法。 选择排序的原理: 一组无序待排数组,做升序
    的头像 发表于 09-25 16:30 835次阅读
    <b class='flag-5'>排序</b><b class='flag-5'>算法</b>之选择<b class='flag-5'>排序</b>

    使用C++ sort函数对vector进行自定义排序

    今天在学一些C++ STL容器,看到sort函数允许自定义排序规则,小小地实操了一下。
    的头像 发表于 07-22 10:12 1029次阅读

    c++之STL算法(三)

    c++之STL算法(三)
    的头像 发表于 07-18 15:00 864次阅读
    <b class='flag-5'>c++</b>之STL<b class='flag-5'>算法</b>(三)

    C++之STL算法(二)

    C++之STL算法(二)
    的头像 发表于 07-18 14:49 634次阅读
    <b class='flag-5'>C++</b>之STL<b class='flag-5'>算法</b>(二)

    FPGA排序-冒泡排序介绍

    排序算法是图像处理中经常使用一种算法,常见的排序算法有插入排序、希尔
    发表于 07-17 10:12 660次阅读
    FPGA<b class='flag-5'>排序</b>-冒泡<b class='flag-5'>排序</b>介绍

    Python实现的常见内部排序算法

    排序算法可以分为内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部
    发表于 07-06 12:35 256次阅读
    Python实现的常见内部<b class='flag-5'>排序</b><b class='flag-5'>算法</b>

    C语言实现十大经典排序算法

    比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破O(nlogn),因此也称为非线性时间比较类排序
    发表于 06-25 10:23 271次阅读
    C语言实现<b class='flag-5'>十大</b>经典<b class='flag-5'>排序</b><b class='flag-5'>算法</b>

    常见排序算法分类

    本文将通过动态演示+代码的形式系统地总结十大经典排序算法排序算法 算法分类 —— 十种常见
    的头像 发表于 06-22 14:49 620次阅读
    常见<b class='flag-5'>排序</b><b class='flag-5'>算法</b>分类

    详解DeepMind排序算法

    DeepMind 的这一发现确实居功至伟,但不幸的是,他们未能解释清楚算法。下面,我们来详细看看他们发布的一段汇编代码,这是一个包含三个元素的数组的排序,我们将伪汇编转换为汇编:
    的头像 发表于 06-21 15:38 251次阅读

    利用强化学习来探索更优排序算法的AI系统

    前言 DeepMind 最近在 Nature 发表了一篇论文 AlphaDev[2, 3],一个利用强化学习来探索更优排序算法的AI系统。 AlphaDev 系统直接从 CPU 汇编指令的层面入手
    的头像 发表于 06-19 10:49 388次阅读
    利用强化学习来探索更优<b class='flag-5'>排序</b><b class='flag-5'>算法</b>的AI系统

    详细介绍8种最常用的排序算法

    在计算机科学领域中,排序算法是一种基本的算法排序算法可以将一个数据集合重新排列成一个按照某种规则有序的集合,常用于数据检索、数据压缩、数据
    的头像 发表于 06-06 14:52 2024次阅读

    C语言经典排序算法总结

    本文将通过动态演示+代码的形式系统地总结十大经典排序算法
    发表于 06-05 10:56 387次阅读
    C语言经典<b class='flag-5'>排序</b><b class='flag-5'>算法</b>总结

    C++入门之通用算法

    C++ 是一种强大的编程语言,它提供了许多通用算法,可以用于各种容器类型。这些算法是通过迭代器来操作容器中的元素,因此它们是通用的,可以用于不同类型的容器。在本篇博客中,我们将详细介绍 C++
    的头像 发表于 05-17 09:40 411次阅读