快速排序(QuickSort) ---JAVA实现 带图详细 适合新手入门

**

快速排序(QuickSort) —JAVA实现 带图详细

适合新手入门

**

快速排序的原理:选择一个关键值作为基准值(一般都选择第一个数字)。比基准值小的都在左边序列(一般是无序的),比基准值大的都在右边(一般是无序的)。

一次循环:从后往前比较,用基准值和最后一个值比较,如果比基准值小的交换位置,如果没有继续比较下一个,直到找到第一个比基准值小的值才交换。找到这个值之后,又从前往后开始比较,如果有比基准值大的,交换位置,如果没有继续比较下一个,直到找到第一个比基准值大的值才交换。

上图上例子:(6 1 2 7 9 3 4 5 10 8)

(一)分别从初始序列“6 1 2 7 9 3 4 5 10 8”两端开始“探测”。选择6作为基准值,先从右往左找一个小于6的数,再从左往右找一个大于6的数,然后交换他们。这里可以用两个变量i和j,分别指向序列最左边和最右边。我们为这两个变量起个好听的名字“哨兵i”和“哨兵j”。刚开始的时候让哨兵i指向序列的最左边(即i=1),指向数字6。让哨兵j指向序列的最右边(即=10),指向数字。
在这里插入图片描述

(二)首先哨兵j开始出动。因为此处设置的基准数是最左边的数,所以需要让哨兵j先出动,这一点非常重要。哨兵j一步一步地向左挪动(即j–),直到找到一个小于6的数停下来。接下来哨兵i再一步一步向右挪动(即i++),直到找到一个数大于6的数停下来。最后哨兵j停在了数字5面前,哨兵i停在了数字7面前。
在这里插入图片描述

(三)现在交换哨兵i和哨兵j所指向的元素的值。交换之后的序列如下:

6 1 2 5 9 3 4 7 10 8
在这里插入图片描述

(四)第一次交换结束。接下来开始哨兵j继续向左挪动(再友情提醒,每次必须是哨兵j先出发)。他发现了4(比基准数6要小,满足要求)之后停了下来。哨兵i也继续向右挪动的,他发现了9(比基准数6要大,满足要求)之后停了下来。此时再次进行交换,交换之后的序列如下:

6 1 2 5 4 3 9 7 10 8
在这里插入图片描述
在这里插入图片描述

(五)第二次交换结束,“探测”继续。哨兵j继续向左挪动,他发现了3(比基准数6要小,满足要求)之后又停了下来。哨兵i继续向右移动,糟啦!此时哨兵i和哨兵j相遇了,哨兵i和哨兵j都走到3面前。说明此时“探测”结束。我们将基准数6和3进行交换。交换之后的序列如下:

3 1 2 5 4 6 9 7 10 8
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

第一轮“探测”真正结束。此时以基准数6为分界点,6左边的数都小于等于6,6右边的数都大于等于6。回顾一下刚才的过程,其实哨兵j的使命就是要找小于基准数的数,而哨兵i的使命就是要找大于基准数的数,直到i和j碰头为止。

现在,以基准数6为分界,左面是3 1 2 5 4,右面是9 7 10 8.分别处理左右两面。
先看3 1 2 5 4:以3为基准,从右往左面扫,遇到2(j位置所对应的数字),从左往右扫,i遇到了j,则交换基准3和2的位置,交换后的序列为:2 1 3 5 4.

3已经在它应该在的位置上了,看3左面的序列:2 1 ,以2为基准进行调整
调整后为1 2 。看3右面的序列:5 4 ,以5为基准进行调整,调整后为4 5 。
此时的序列为:1 2 3 4 5 6 9 7 10 8
6右面的序列用同样的方法进行调整,调整后的序列为:1 2 3 4 5 6 7 8 9 10.

代码实现:

package sort;

public class QuickSort {
	public static void quick_1(int []arr,int low,int high){
		int i,j,temp;
		if(low > high){
			return;
		}
		
		i = low;//左边哨兵的索引
		j = high;//右边哨兵的索引
		temp = arr[low];//temp是基准位
		
		while(i < j){
			//从右面往左面扫描
			while(temp <= arr[j] && i < j){
				j--;
			}
			//从左面往右面扫
			while(temp >= arr[i] && i < j){
				i++;
			}
			//满足条件则交换
			if(i < j){
				int z,y;
				//z、y是临时参数,用于存放左右哨兵 所在位置的数据
				z = arr[i];
				y = arr[j];
				
				//交换
				arr[i] = y;
				arr[j] = z;
		}	
	}
		//跳出i<j,说明i=j
		arr[low] = arr[i];
		arr[i] = temp;
		
		//递归调用左半数组
		quick_1(arr,low,j-1);
		//递归调用右半数组
		quick_1(arr,j+1,high);
	}
	public static void main(String[] args) {
		int[] arr = {6,1,2,7,9,3,4,5,10,8};
		quick_1(arr,0,arr.length - 1);
		for (int i = 0; i < arr.length; i++) {
			System.out.print(arr[i]+" ");
		}
	}
}

快排比冒泡排序快的原因??
相比冒泡排序,每次交换是跳跃式的。每次排序的时候设置一个基准点,将小于等于基准点的数全部放到基准点的左边,将大于等于基准点的数全部放到基准点的右边。这样在每次交换的时候就不会像冒泡排序一样每次只能在相邻的数之间进行交换,交换的距离就大的多了。因此总的比较和交换次数就少了,速度自然就提高了。

快排时间复杂度:
O(NlogN)

最坏情况时间复杂度:
O(N2) 与冒泡排序相同

原文地址:http://developer.51cto.com/art/201403/430986.htm

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值