热门标签 | HotTags
当前位置:  开发笔记 > IOS > 正文

C/C++实现双路快速排序算法原理

这篇文章主要为大家详细介绍了CC++实现双路快速排序算法原理,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了C/C++实现双路快速排序算法的具体代码,供大家参考,具体内容如下

看了刘宇波的视频,讲双路快速排序的,原理讲的很直观,程序讲解也一看就懂。这里写一下自己的理解过程,也加深一下自己的理解。

首先说一下为什么需要双路排序,在有些带有许多重复数据的数组里,使用随机快速排序或者最简单的快速排序算法时,由于重复的数据会放在原来的索引位置不动,就回导致划分数组时划分的某一部分太长,起不到分段排序的效果,这样就导致算法退化成O(n^2)的复杂度。就像下图:

为了解决这个问题,双路快速排序采用的方法是对等于v的数也进行交换,原理如下所述:

首先选择一个数作为标志,放在数组的最左侧,下标为l,在数组左边放小于v的数,右侧放大于v的数。
之后,先从l+1开始遍历数组,当数据小于v时,该数据属于左侧橙色部分,保持其位置不动,i++,继续向后遍历,当找到某个数大于或者等于(注意,这里等于很重要)v时,停止遍历。转而开始根据j来遍历数组,j不断减小,索引数组的数据,当索引到某个数小于或者等于v时,停止遍历。如下图所示:

这时两个绿色的区域就是分别属于v的部分,而i,j所对应的索引数据要交换位置。

之后,将i,j分别向后向前移动一位,继续开始新的索引,直到i和j重合或者i>j位置,就完成了partition的过程。

下面贴出代码:

主函数 main.cpp

// QuickSort2.cpp : 双路快速排序,适用于解决有很多重复数据的数组。
//

#include "stdafx.h"
#include "E:/学习/C++/数据结构和算法/code/算法/排序算法/common/sortTestHelper.h"
#include "QuickSort.h"
#include "RadomQuickSort.h"
#include "QuickSort2.h"

using namespace std;

int main()
{
 int n = 100000;
 int *arr1 = SortTestHelper::generateRadomArray(n, 0, 50);
 int *arr2 = SortTestHelper::generateRadomArray(n, 0, 50);
 int *arr3 = SortTestHelper::generateRadomArray(n, 0,50);
 SortTestHelper::sortTime("随机快速排序", RadomQuickSort, arr1, n);
 SortTestHelper::sortTime("快速排序", QuickSort, arr2, n);
 SortTestHelper::sortTime("双路快速排序", QuickSort2, arr3, n);
 delete[] arr1;
 delete[] arr2;
 delete[] arr3;
 return 0;
}

双路快速排序算法 QuickSort2.h

#ifndef QUICKSORT2_H
#define QUICKSORT2_H

#include 
#include 
using namespace std;

template 
int __partition3(T *arr, int l, int r)
{
//此处结合随机快速排序的算法进行了优化,标记点在数组里随机选择
 int RAND = (rand() % (r - l + 1) + l);
 swap(arr[RAND], arr[l]);

 int v = arr[l];
 int i = l + 1;
 int j = r;
 while (true)
 {
 while (i <= r&&arr[i] = l + 1 && arr[j] > v) j--;
 if (i > j)
 {
 break;
 }
 swap(arr[i], arr[j]);
 i++;
 j--;
 }
 swap(arr[l], arr[j]);
 return j;
}

template 
void __QuickSort2(T *arr,int l,int r)
{
 if (l>=r)
 {
 return;
 }
 int p = __partition3(arr, l, r);
 __QuickSort2(arr, l, p - 1);
 __QuickSort2(arr, p + 1, r);
}

template 
void QuickSort2(T *arr, int n)
{
 __QuickSort2(arr, 0,n-1);
}


#endif

随机快速排序 RadomQuickSort.h

#ifndef RADOMQUICKSORT_H
#define RADOMQUICKSORT_H

#include 
#include 

using namespace std;

template 
int __Randpartition(T *arr, int l, int r)
{
 //选择开头的数作为分割的数
 int RAND = arr[rand() % (r - l + 1) + l];
 swap(arr[l], RAND);
 int i = arr[l];
 //遍历数组,使得arr[l,l+1,...j]arr[l]
 int j = l;
 //如果当前数据大于arr[l],就无需改变位置,如果小于arr[l],就将当前数据与分割点的数据后一个数据交换
 for (size_t k = j + 1; k <= r; k++)
 {
 if (arr[k]
void __RadomQuickSort(T *arr, int l, int r)
{
 if (l >= r)
 {
 return;
 }
 int p = __Randpartition(arr, l, r);
 __RadomQuickSort(arr, l, p - 1);
 __RadomQuickSort(arr, p + 1, r);
}

template 
void RadomQuickSort(T *arr, int n)
{
 __RadomQuickSort(arr, 0, n - 1);
}

#endif

快速排序 QuickSort.h

#ifndef QUICKSORT_H
#define QUICKSORT_H

using namespace std;

template 
int __partition(T *arr, int l, int r)
{
 //选择开头的数作为分割的数
 int i = arr[l];
 //遍历数组,使得arr[l,l+1,...j]arr[l]
 int j = l;
 //如果当前数据大于arr[l],就无需改变位置,如果小于arr[l],就将当前数据与分割点的数据后一个数据交换
 for (size_t k = j + 1; k <= r; k++)
 {
 if (arr[k]
void __QuickSort(T *arr, int l, int r)
{
 if (l >= r)
 {
 return;
 }
 int p = __partition(arr, l, r);
 __QuickSort(arr, l, p - 1);
 __QuickSort(arr, p + 1, r);
}

template 
void QuickSort(T *arr, int n)
{
 __QuickSort(arr, 0, n - 1);
}

#endif

SortTestHelper 函数

#ifndef SORTTESTHELPER_H
#define SORTTESTHELPER_H

#include 
#include 
#include 
#include 

using namespace std;

namespace SortTestHelper 
{
//产生一个从[rangeL,rangeH]的随机数组,数组个数是n
 int* generateRadomArray(int n,int rangeL,int rangeH)
 {
 //为了算法的健壮性,需要判断错误输入
 assert(rangeL 
 void printArr(T *arr,int n)
 {
 for (size_t i = 0; i 
 bool ifSort(T *arr,int n)
 {
 for (size_t i = 0; i arr[i+1])
 {
 return false;
 }
 }
 return true;
 }

//计算程序运行时间
 template
 //函数输入参数是:所需要计算的运行的函数的名称,函数的指针,函数的输入数组,输入数组的个数
 void sortTime(string funName,void(*sort)(T*arr, int), T* arr,int n)
 {
 clock_t startime = clock();
 sort(arr,n);
 clock_t endtime = clock();

 assert(ifSort(arr, n));
 std::cout <

最终结果三种算法对10万个具有重复的数据的排序时间如下:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 回文字串(Manacher算法介绍)
    Manacher算法(原理)分i<P与i>P的两种情况,P为右端边界+1,即不在回文字串中(废话)一。i>P,就以i为中心,重新找回文,下文while的 ... [详细]
  • 上机目的:学会使用循环控制语句解决实际问题上机内容:编写多分支选择结构程序,根据个人月收入总额,计算出应缴税款和税后收入。**Copyright(c)2012,烟台大学计算机学院*A ... [详细]
  • 一键重装系统哪个干净?2022最干净的一键重装系统推荐
    一键重装系统哪个干净?目前市面上有很多一键重装系统的软件,去除了各种繁琐的操作步骤,操作简单上手容易轻轻一点就能完成,新手小白也可以实现一键快速重装,下面小编就给大家推荐几款最干净 ... [详细]
  • HDU3699AhardAoshuProblem(暴 ... [详细]
  • 条款18:让接口容易被正确使用,不容易被误用1、许多客户端错误可以因为导入新类型而获得预防。在防范“不值得拥有的代码”上,类型系统是你的主 ... [详细]
  • iPhone BootROM 漏洞说明及威胁评估
    iPhone BootROM 漏洞说明及威胁评估 ... [详细]
  • datasnap的多表更新
    补课很多,xe一路走来,提供了许许多多的多层数据处理的方法。确实已经提供数个最优方案。把下面的图跑通了,也就解决核心问题了。【Berlindatasnap开发手册p ... [详细]
  • 物联网概论(IoT)__Chp13 体系架构、射频识别(RFID)
    chp1概述物联网(InternetofThings)IoT物联网的定义是,通过射频识别、红外感应器、全球定位系统、激光扫描器等信息传感设备,按照约定的协议,把任何物品与互联网连接 ... [详细]
  • 并非所有孤独症患者,都是天赋异禀
    “神经多样性运动的动机是好的,但是它关切的只有高功能人群,而忽视了那些在重度孤独症中挣扎不得脱身的人。”本文来自微信公众号:神经现实(ID:neureality),作者:Moheb ... [详细]
  • 约瑟夫环(单向链表) 问题描述:有n个小朋友围成一圈玩游戏,小朋友从1至n编号,2号小朋友坐在1号小朋友的顺时 ... [详细]
  • DSO全家桶(二)——DSO前端:提取梯度点
    嗨,各位读者朋友们好!今天我们接着讲DSO全家桶的第二讲——提取梯度点。首先,我们会先介绍一下DSO中从图像读取到提取梯度点的整个流程,然后再对重点的部分——“提取梯度点”进行讲解 ... [详细]
  • PTA刷题Basic篇——1093.字符串A+B——Day(47)
    问题描述输出在两个字符串中仅出现一次的字符,先输出A,在A中输出过的在B中不能再重复出现。 题目分析 开辟一个长度为200的数组a(因为ASCII码值最 ... [详细]
  • 1.最长递增子序列求一段字符串的最长递增子序列问题分析:设序列为:Aa0,a1,a2,a3,a4,a5,,ai,定义D(i)为选i作为序列一项后,后面序列中第i项更 ... [详细]
  • workerman 在windows 下安装使用
    windows下开启进程,如果代码里面有错误,有时候会报出来,linux不会;windows下php代码中如果有exit,会进程被关闭或者直接启动不了;linux则会终止代码执行, ... [详细]
  • 最新发布的前端开发相关工具
    最新发布的前端开发相关工具,Go语言社区,Golang程序员人脉社 ... [详细]
author-avatar
Tong-david
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有