快速排序总结

数据结构之排序算法——快速排序

代码很多地方借鉴了  http://my.csdn.net/MoreWindows 他的思想,

本人认为该作者已经写的很好了,只是在他的基础上加入了一些自己的理解和说明

如果涉及到版权的问题,请联系我的邮箱,我会尽快删除

希尔排序想关链接:

维基百科:https://zh.wikipedia.org/wiki/%E5%BF%AB%E9%80%9F%E6%8E%92%E5%BA%8F#C.E8.AA.9E.E8.A8.80

百度百科:http://baike.baidu.com/view/19016.htm

参考博客 :http://blog.csdn.net/morewindows/article/details/6684558

快速排序 对大量的数据具有很好的性能,

快速排序的基本思想就是,先对数组进行从右向左的扫描,找到第一不大于当前定义的关键值数,与它进行交换,然后在从左向右进行扫描,找到第一个不小于当前关键值的数,将它们交换,然后将数据分成两个部分,前面一部分都小于关键字,后面一部分都大于关键字,然后再前后的部分继续进行前面描述的操作,这就需要递归的调用自己。

源码如下:

void Quick_Sort(int array[], int arrayLen)

{

Sort(array, 0, arrayLen-1);

}

void Sort(int array[],  int low, int height)

{

if (low < height)

{

int retPivotkey = partition(array, low, height);

Sort(array, low, retPivotkey - 1);

Sort(array, retPivotkey + 1, height);

}

}

int partition(int array[], int right, int left)

{

// 将所有的比关键字小的元素,移动到关键字的左边,比关键字大的元素移动到关键字的右边

int pivotkey = array[right];

while (right < left)

{

while (right < left && array[left] >= pivotkey)

-- left;

if (right < left)

{

array[right++] = array[left];

}

while (right < left && array[right] < pivotkey)

++ right;

if (right < left)

{

array[left--]=array[right] ;

}

}

// 将关键字插入想关的位置

array[right] = pivotkey;

// 返回关键字的位置,便于下次循环

return right;

}

源码这样写是为了和前面排序算法相互统一,我也看到了一个合并的写法

void Quick_Sort(int array[], int low, int height)

{

// 将low 和 height  缓存 为后面的递归的实现提供界限

int right = low, left = height;

if (low < height)

{

int pivotkey = array[right];

while (right < left)

{

while (right < left && array[left] >= pivotkey)

--left;

if (right < left)

{

array[right++] = array[left];

}

while (right < left && array[right] < pivotkey)

++right;

if (right < left)

{

array[left--] = array[right];

}

}

// 将关键字插入想关的位置

array[right] = pivotkey;

Quick_Sort(array, low, right - 1);

Quick_Sort(array, right + 1, height);

}

}

在查资料的时候也看到了一个迭代的版本,这个版本来自维基百科

维基百科链接: https://zh.wikipedia.org/wiki/%E5%BF%AB%E9%80%9F%E6%8E%92%E5%BA%8F#C.E8.AA.9E.E8.A8.80

源码如下:

typedef struct _Range {

int start, end;

} Range;

Range new_Range(int s, int e) {

Range r;

r.start = s;

r.end = e;

return r;

}

void swap(int *x, int *y) {

int t = *x;

*x = *y;

*y = t;

}

void quick_sort2(int arr[], const int len) {

if (len <= 0)

return; // 避免len为负值时出错

// r[]模拟栈 ,p为栈中元素的数量, r[p++] 为push, r[--p] 为pop且 取得元素

Range *r= (Range *)malloc(sizeof(Range)*len);// 因为vs 不支持c99 所以,只能动态开辟了,支持c99的可以使用 Range r[len];

if (r == NULL)

return;

int p = 0;

// 模拟栈的入栈

r[p++] = new_Range(0, len - 1);

while (p)

{

// 模拟栈的出栈

Range range = r[--p];

// 当start大于end时,结束当前循环

if (range.start >= range.end)

continue;

int mid = arr[range.end];

int left = range.start, right = range.end - 1;

while (left < right)

{

// 找到第一个不小于mid的元素

while (arr[left] < mid && left < right)

left++;

// 找到第一个不大于mid的元素

while (arr[right] >= mid && left < right)

right--;

swap(&arr[left], &arr[right]);

}

// 避免数组元素越界

if (arr[left] >= arr[range.end])

{

swap(&arr[left], &arr[range.end]);

}

else

{

// 如果left 指针没有越界

left++;

}

// 将左右两个部分的范围入栈

r[p++] = new_Range(range.start, left - 1);

r[p++] = new_Range(left + 1, range.end);

}

free(r);

}

时间: 2024-08-14 11:30:25

快速排序总结的相关文章

快速排序

快速排序的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列. 快速排序是一种不稳定的排序算法,也就是说,多个相同的值的相对位置也许会在算法结束时产生变动 快速排序是C.R.A.Hoare于1962年提出的一种划分交换排序.它采用了一种分治的策略,通常称其为分治法(Divide-and-ConquerMethod). 该方法的基本思想是:

快速排序——Python

快速排序: 在一组数据中选择一个基准值,让后将数据分为两个部分,一部分大于基准,一部分小于基准,然后按此方法将两个部分分组,直到不能再分为止. 需要明白一个概念递归和分而治之的概念. Python实现: 1 # 快速排序 2 3 import random 4 5 def quick_sort(arr): 6 # 边界条件 7 if len(arr) < 2: 8 return arr 9 key = random.choice(arr) # 选择基准 10 left = [i for i in

快速排序的实现(不保证效率

众所周知,快速排序的核心是分治的思想,选一个基准出来,然后通过划分操作,使得,该元素最终处于的位置的左边的元素都小于等于它,右边的元素都大于等于它 划分操作就是两次递归嘛,没什么的,关键在于不借助外部空间我们如何实现划分操作 首先我们不知道该元素放在哪里,显然这是最后才能确定的, 我了解到一种填坑法的实现... 那就是首先保存第一个位置的值,然后从后向前扫描第一个小于x的值,我们就可以直接覆盖第一个位置的值,然后我们再从前向后找大于x的值, 把后面的坑填上 下面枚举几种情况 基准前后有相同数量的

快速排序的总结

快速排序的思想是分而治之,利用递归达到快速排序的效果 首先要选定一个基准数,一般选择最左边的数为基准数,排序的目标就是让这个基准数的左边全小于这个基准数,右边全大于这个基准数.然后以这个基准数为分隔线,在左右两侧再次调用这个排序的函数,直到全部有序.简述过程: 以  8 9 4 7 2 6 首选 1. 选择两个哨兵 i,j 分别指向8,6,基准数为8 2.从j哨兵开始,因为j指向的6小于基准数8,不符合j指向的数都要大于8的要求,所以将j指向的数覆盖i指向的数,同时i指向的数变成9 6 9 4

[数据结构] 快速排序

基本思想 快速排序(Quicksort)是对冒泡排序的一种改进,又称划分交换排序(partition-exchange sort. 快速排序使用分治法(Divide and conquer)策略来把一个序列(list)分为两个子序列(sub-lists). 步骤为: ①.从数列中挑出一个元素,称为"基准"(pivot) ②.重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边).在这个分区结束之后,该基准就处于数列的中间位置.这个称

使用JAVA直观感受快速排序与冒泡排序的性能差异

初学算法,肯定会编写排序算法 其中两个最为有名的就是冒泡排序和快速排序 理论上冒泡排序的时间复杂度为O(N^2),快速排序的时间复杂度为O(NlogN) 下面本门使用JAVA,分别编写三段排序程序 对十万个0-9999的整数进行一次冒泡排序 对十万个0-9999的整数进行1000次快速排序,使用递归完成 对十万个0-9999的整数进行1000次快速排序,使用堆栈完成 对十万个0-9999的整数进行一次冒泡排序: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

Swift实现的快速排序及sorted方法的对比

Swift语言有着优秀的函数式编程能力,面试的时候面试官都喜欢问我们快速排序,那么用Swift如何实现一个快速排序呢?首先扩展Array类: extension Array { var decompose : (head: T, tail: [T])? { return (count > 0) ? (self[0], Array(self[1..<count])) : nil } } 属性decompose的作用是返回数组中的第一个元素和剩下的元素,注意这个属性是可选型的,当count为0的时

Java实现排序算法之快速排序

一.综述 快速排序是交换排序中的一种,平均算法复杂度是O(nlogn),最坏O(n*n).下面用Java实现一个快速排序,并用注释的方式解释了思想和原理. 二.Java实现堆排序 三.结果检验 版权声明:本文为博主原创文章,未经博主允许不得转载.

单链表排序——快速排序实现

利用快速排序,同向一前一后两个指针 #ifndef LIST_H_ #define LIST_H_ #include <iostream> #include <utility> class List { private: struct ListNode { int _value; ListNode* _next; }; public: List(): _head(nullptr) {} ~List() { while (nullptr != _head) { auto tmp =

排序之快速排序

package com.xsz.demo; /**     * @author cwqi    * @date 2015-1-6    */ //排序之快速排序,基本思路為分治+挖坑,方法有兩種:一是雙邊掃面:二是單邊掃描. public class QuickSort { public static void main (String[] args){ int a[] ={2,3,30,1,4,56,2,7,3,8}; //int a[] ={4,5,6,2,7,3,8}; //int a[]