partition函数是快排的核心部分

它的目的就是将数组划分为<=pivot和>pivot两部分,或者是<pivot和>=pivot

其实现方法大体有两种,单向扫描版本双向扫描版本,但是具体到某个版本,其实现方法也是千差万别,参差不齐。本着严谨治学的态度,我将目前所接触的所有实现列举出来,并作出比较。除了伪代码,我也会给出相应的C&C++实现,供读者参考。

单向扫描:

下面是算法导论中例子

PARTITION(A, p, r)
x = A[r]
i = p -
for j = p to r -
if A[j] <= x
i = i +
exchange A[i] with A[j]
exchange A[i + ] with A[r]
return i + int partition(int a[], int p, int r)
{
int x = a[r];
int i = p - ;
int j = p;
for (; j < r; ++j)
if (a[j] <= x)
swap(&a[++i], &a[j]);
swap(&a[i + ], &a[j]);
return i + ;
}

这个是标准的单向扫描,其思路是:

将小于或等于pivot的元素通过交换全部移到前面去,这里需要注意的是i的作用,这是个哨兵,用于记录交换后的位置,也就是i之前的元素都是交换好了的。

下面是一些可以变动的地方:

1.可以将小于pivot的元素移到前面去,而不是小于等于,这样可以减少些交换次数,同理,可以将大于pivot的元素移到后面去,不过这样就需要倒序遍历了

2.或者是将i的初始值设置为p,而不是p-1;

3.可以将pivot设置成第一个元素;

4.存在i=j的情况,这时候的交换就是多余的,可以优化掉。

下面是稍作优化的版本

int partition(int a[], int p, int r)
{
int x = a[r];
int i = p;
int j = p;
for (; j < r; ++j)
if (a[j] < x) {
if (i != j)
swap(&a[i], &a[j]);
i++;
}
swap(&a[i], &a[j]);
return i;
}

双向扫描:

算法导论上的课后题有该算法,但是错误百出,这里以《算法》第四版的方法为例

PARTITION(A, p, r)
x = A[p]
i = p
j = r +
while true
repeat
j = j -
until A[j] <= x
repeat
i = i +
until A[i] >= x
if i >= j
break
exchange A[i] with A[j]
exchange A[p] with A[j]
return j int partition(int a[], int p, int r)
{
int x = a[p];
int i = p;
int j = r + ;
while (true) {
while (a[--j] > x);
while (a[++i] < x);
if (i >= j)
break;
swap(&a[i], &a[j]);
}
swap(&a[j], &a[p]);
return j;
}

其思路是从左到右找到大于等于pivot的元素,从右到左找到小于等于pivot的元素,然后将这两个元素交换,直到左右扫描相遇,最后还要进行一次交换,将pivot调整到正确位置

这是上面程序的变种,看起来差别很大,不过原理是相同的

int partition(int a[], int p, int r)
{
int x = a[p];
int i = p + ;
int j = r;
while (i <= j) {
while (a[j] > x) j--;
while (a[i] < x) i++;
if (i >= j)
break;
swap(&a[i++], &a[j--]);
}
swap(&a[j], &a[p]);
return j;
}

我们看一下它的扫描条件,一个是大于等于,一个是小于等于,也就是说左右扫描点存在都等于pivot的情况,这时候我们是不用交换的。根据互补原理,一个扫描点条件是大于等于,那么另一扫描点条件应该是互补条件小于,这样两个扫描点交换就不会出现交换相等元素的情况。

另外程序还存在着巨大的溢出漏洞,内层的while循环如:

while (a[i] < x) i++;

我们无法保证其不会越界,事实上,我经过测试,发现i的值一旦越界就不确定了,虽然都能保证i >= j的临界条件,但我们还是应该尽量避免越界问题

可以在循环中加入越界条件

int partition(int a[], int p, int r)
{
int x = a[p];
int i = p;
int j = r + ;
while (true) {
while (i < j && a[--j] >= x);
if (i >= j) break;
while (i < j && a[++i] < x);
if (i >= j) break;
swap(&a[i], &a[j]);
}
swap(&a[j], &a[p]);
return j;
}

变种的防越界版如下

int partition(int a[], int p, int r)
{
int x = a[p];
int i = p + ;
int j = r;
while (true) {
while (i <= j && a[j] >= x) j--;
if (i > j) break;
while (i <= j && a[i] < x) i++;
if (i > j) break;
swap(&a[i++], &a[j--]);
}
swap(&a[j], &a[p]);
return j;
}

左右扫描的版本还有很多,让我们再来举几个例子

网上流传比较广的一个版本是下面这个

int partition(int a[], int p, int r)
{
int x = a[p];
int i = p;
int j = r;
while (i < j)
{
while (i < j && a[j] >= x) j--;
if (i >= j) break;
a[i++] = a[j];
while (i < j && a[i] < x) i++;
if (i >= j) break;
a[j--] = a[i];
}
a[i] = x;
return i;
}

仔细观察会发现,它与我们上面介绍的版本几乎如出一辙,不同的是,它没有使用swap交换元素,而是依次覆盖,最后再把pivot归位

具体过程可以参阅:http://blog.csdn.net/morewindows/article/details/6684558

算法的时间复杂度是O(n),但是为什么要写成双循环呢?我们完全可以把它改成单循环,代码如下:

int partition(int a[], int p, int r)
{
int x = a[p];
int i = p + ;
int j = r;
while (i <= j) {
if (a[j] > x)
{
j--;
continue;
}
if (a[i] < x)
{
i++;
continue;
}
swap(&a[i++], &a[j--]);
}
swap(&a[j], &a[p]);
return j;
}

但是,并不推荐这种做法,因为每次判断i的时候,势必会再次判断j,多一次比较。

总结:个人推荐单向扫描的优化版本,双向扫描可以看到会有越界的问题,为了防止越界付出了一定代价。

快速排序 partition函数的所有版本比较的更多相关文章

  1. 剑指Offer28 最小的K个数(Partition函数应用+大顶堆)

    包含了Partition函数的多种用法 以及大顶堆操作 /*********************************************************************** ...

  2. 寻找序列中最小的第N个元素(partition函数实现)

    Partition为分割算法,用于将一个序列a[n]分为三部分:a[n]中大于某一元素x的部分,等于x的部分和小于x的部分. Partition程序如下: long Partition (long a ...

  3. Partition函数

    快排中核心的方法应该算是Partition函数了,它的作用就是将整个数组分成小于基准值的左边,和大于基准值的右边. 普通的Partition函数是这样的: public static int part ...

  4. 字符串的partition函数

    partition函数 str1='sdga2a34'aa=str1.partition('a') print(aa) """ ('sdg', 'a', '2a34') ...

  5. c++多线程编程:实现标准库accumulate函数的并行计算版本

    今天使用c++实现了标准库头文件<numeric>中的accumulate函数的并行计算版本,代码如下,注释写的比较详细,仅对其中几点进行描述: ①该实现假定不发生任何异常,故没有对可能产 ...

  6. 快速排序中的partition函数的枢纽元选择,代码细节,以及其标准实现

    很多笔试面试都喜欢考察快排,叫你手写一个也不是啥事.我很早之前就学了这个,对快速排序的过程是很清楚的.但是最近自己尝试手写,发现之前对算法的细节把握不够精准,很多地方甚至只是大脑中的一个映像,而没有理 ...

  7. 快速排序的Partition函数

    1 //数组中两个数的交换 2 static void swap(int[] nums, int pos1, int pos2){ 3 int temp = nums[pos1]; 4 nums[po ...

  8. 快速排序partition过程常见的两种写法+快速排序非递归实现

    这里不详细说明快速排序的原理,具体可参考here 快速排序主要是partition的过程,partition最常用有以下两种写法 第一种: int mypartition(vector<int& ...

  9. C++关于strcpy等函数的安全版本

    如下程序: #include <iostream> using namespace std; int main() { ]; strcpy(ch1,"); } 在VS2012上面 ...

随机推荐

  1. oracle取字符串长度的函数length()和hengthb()

    http://blog.itpub.net/161195/viewspace-613263/ lengthb(string)计算string所占的字节长度 :返回字符串的长度,单位是字节 length ...

  2. Xshell利用登录脚本从服务器登录到另外一个服务器

    欢迎和大家交流技术相关问题: 邮箱: jiangxinnju@163.com 博客园地址: http://www.cnblogs.com/jiangxinnju GitHub地址: https://g ...

  3. node Express安装与使用(一)

    首先放上官网地址 http://www.expressjs.com.cn/ 学会查阅官方手册,它是最好的资料. 1.Express安装 首先确定你已经安装了 Node.js,然后去你创建的项目目录下( ...

  4. 都能读懂的css3 3D变形效果

    css3 3D变形效果 CSS3 transform3D变形 transform的含义是:改变,使-变形:转换 三维变换使用基于二维变换的相同属性,如果您熟悉二维变换,你们发现3D变形的功能和2D变换 ...

  5. .Net程序员学用Oracle系列(16):访问数据库(ODP.NET)

    1..Net for Oracle 常见数据库驱动 1.1.微软提供的驱动 1.2.甲骨文提供的驱动 1.3.其它厂商提供的驱动 2.ODP.NET 常见问题分析 2.1.参数化问题 2.2.方法调用 ...

  6. LINQ查询表达式和LAMBDA点标记方法基础

    在上一篇文章中,我们介绍了LINQ的一些基本用法,这一篇我们来看下另一种更简洁更优雅的表达式,Lambda表达式,也可以叫做点标记方法. 相信大家在实际工作中都用到过这两种方式,下面我们还是用实例来看 ...

  7. 1.使用SignalR实现页面即时刷新(服务端主动推送)

    模块功能说明: 实现技术:sqlserver,MVC,WebAPI,ADO.NET,SignalR(服务器主动推送) 特殊车辆管理--->移动客户端采集数据存入数据库---->只要数据库数 ...

  8. ABP入门系列(11)——编写单元测试

    ABP入门系列目录--学习Abp框架之实操演练 源码路径:Github-LearningMpaAbp 1. 前言 In computer programming, unit testing is a ...

  9. [Hadoop] - Win7下提交job到集群上去

    一般我们采用win开发+linux hadoop集群的方式进行开发,使用插件:hadoop-***-eclipse-plugin. 运行程序的时候,我们一般采用run as application或者 ...

  10. JavaScript中国象棋程序(8) - 进一步优化

    在这最后一节,我们的主要工作是使用开局库.对根节点的搜索分离出来.以及引入PVS(Principal Variation Search,)主要变例搜索. 8.1.开局库 这一节我们引入book.js文 ...