快速排序

基本思想:
通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
步骤分析:
假设用户输入了如下数组:

下标    0	1	2	3	4	5
数据	6	2	7	3	8	9

创建变量i=0(指向第一个数据), j=5(指向最后一个数据), k=6(赋值为第一个数据的值)。
我们要把所有比k小的数移动到k的左面,所以我们可以开始寻找比6小的数,从j开始,从右往左找,不断递减变量j的值,我们找到第一个下标3的数据比6小,于是把数据3移到下标0的位置,把下标0的数据6移到下标3,完成第一次比较:

下标	0	1	2	3	4	5
数据	3	2	7	6	8	9

i=0 j=3 k=6
接着,开始第二次比较,这次要变成找比k大的了,而且要从前往后找了。递加变量i,发现下标2的数据是第一个比k大的,于是用下标2的数据7和j指向的下标3的数据的6做交换,数据状态变成下表:

下标	0	1	2	3	4	5
数据	3	2	6	7	8	9

i=2 j=3 k=6
称上面两次比较为一个循环。
接着,再递减变量j,不断重复进行上面的循环比较。
在本例中,我们进行一次循环,就发现i和j“碰头”了:他们都指向了下标2。于是,第一遍比较结束。得到结果如下,凡是k(=6)左边的数都比它小,凡是k右边的数都比它大:

下标	0	1	2	3	4	5
数据	3	2	6	7	8	9

如果i和j没有碰头的话,就递加i找大的,还没有,就再递减j找小的,如此反复,不断循环。注意判断和寻找是同时进行的。
然后,对k两边的数据,再分组分别进行上述的过程,直到不能再分组为止。
注意:第一遍快速排序不会直接得到最终结果,只会把比k大和比k小的数分到k的两边。为了得到最后结果,需要再次对下标2两边的数组分别执行此步骤,然后再分解数组,直到数组不能再分解为止(只有一个数据),才能得到正确结果。
代码实现(Java)

package sort;

public class QuickSort {
	public static void main(String[] args) {
		int [] data = {1,9,3,5,6,7,569,4,11,9585,66,56};
		for (int i = 0; i < data.length; i++) {
			System.out.print(data[i]+" ");
		}
		quickSort(data);
		System.out.println();
		for (int i = 0; i < data.length; i++) {
			System.out.print(data[i]+" ");
		}
	}

	private static void quickSort(int[] data) {
		// TODO Auto-generated method stub
		if(data.length>0) {
			quickSortCore(data,0,data.length-1);
		}
	}

	private static void quickSortCore(int[] data, int low, int high) {
		// TODO Auto-generated method stub
		
		//1.找到算法出口
		if(low>high) {
			return;
		}
		//2.cun
		int i =low;
		int j = high;
		//3.key
		int key = data[low];
		//4.完成一趟排序
		while(i<j) {
			//4.1 从右往左找到第一个小于key的数
			while(i<j&&data[j]>key) {
				j--;
			}
			//4.2从右往左找到第一个大于key的数
			while(i<j&&data[i]<=key) {
				i++;
			}
			//4.3交换
			if(i<j) {
				int p =data[i];
				data[i]=data[j];
				data[j]=p;
			}
			
		}
		
		int temp = data[i];
		data[i]=data[low];
		data[low]=temp;
		quickSortCore(data, low, i-1);
		quickSortCore(data, i+1, high);
		
	}
	
}

运行效果

1 9 3 5 6 7 569 4 11 9585 66 56 
1 3 4 5 6 7 9 11 56 66 569 9585
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值