`

常用的排序算法的Java实现

 
阅读更多

 

 

常用排序算法小记

排序算法很多地方都会用到,近期又重新看了一遍算法,并自己简单地实现了一遍,特此记录下来,为以后复习留点材料。

废话不多说,下面逐一看看经典的排序算法:

 

1. 选择排序

选择排序的基本思想是遍历数组的过程中,以 i 代表当前需要排序的序号,则需要在剩余的 [i…n-1] 中找出其中的最小值,然后将找到的最小值与 i 指向的值进行交换。因为每一趟确定元素的过程中都会有一个选择最大值的子流程,所以人们形象地称之为选择排序。

举个实例来看看:

初始: [38, 17, 16, 16, 7, 31, 39, 32, 2, 11]

i = 0:  [2 , 17, 16, 16, 7, 31, 39, 32, 38 , 11] (0th [38]<->8th [2])

i = 1:  [2, 7 , 16, 16, 17 , 31, 39, 32, 38, 11] (1st [38]<->4th [17])

i = 2:  [2, 7, 11 , 16, 17, 31, 39, 32, 38, 16 ] (2nd [11]<->9th [16])

i = 3:  [2, 7, 11, 16, 17, 31, 39, 32, 38, 16] ( 无需交换 )

i = 4:  [2, 7, 11, 16, 16 , 31, 39, 32, 38, 17 ] (4th [17]<->9th [16])

i = 5:  [2, 7, 11, 16, 16, 17 , 39, 32, 38, 31 ] (5th [31]<->9th [17])

i = 6:  [2, 7, 11, 16, 16, 17, 31 , 32, 38, 39 ] (6th [39]<->9th [31])

i = 7:  [2, 7, 11, 16, 16, 17, 31, 32, 38, 39] ( 无需交换 )

i = 8:  [2, 7, 11, 16, 16, 17, 31, 32, 38, 39] ( 无需交换 )

i = 9:  [2, 7, 11, 16, 16, 17, 31, 32, 38, 39] ( 无需交换 )

由例子可以看出,选择排序随着排序的进行( i 逐渐增大),比较的次数会越来越少,但是不论数组初始是否有序,选择排序都会从 i 至数组末尾进行一次选择比较,所以给定长度的数组,选择排序的比较次数是固定的: 1 + 2 + 3 + …. + n = n * (n + 1) / 2 ,而交换的次数则跟初始数组的顺序有关,如果初始数组顺序为随机,则在最坏情况下,数组元素将会交换 n 次,最好的情况下则可能 0 次(数组本身即为有序)。

由此可以推出,选择排序的时间复杂度和空间复杂度分别为 O(n2 ) O(1) (选择排序只需要一个额外空间用于数组元素交换)。

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Selection Sorting  
  3.  */  
  4. SELECTION(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         int len = array.length;   
  7.         for (int i = 0; i < len; i++) {   
  8.             int selected = i;   
  9.             for (int j = i + 1; j < len; j++) {   
  10.                 int compare = array[j].compareTo(array[selected]);   
  11.                 if (compare != 0 && compare < 0 == ascend) {   
  12.                     selected = j;   
  13.                 }   
  14.             }   
  15.   
  16.             exchange(array, i, selected);   
  17.         }   
  18.     }   
  19. })  
	/**
	 * Selection Sorting
	 */
	SELECTION(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			int len = array.length;
			for (int i = 0; i < len; i++) {
				int selected = i;
				for (int j = i + 1; j < len; j++) {
					int compare = array[j].compareTo(array[selected]);
					if (compare != 0 && compare < 0 == ascend) {
						selected = j;
					}
				}

				exchange(array, i, selected);
			}
		}
	})

 

 

2. 插入排序

插入排序的基本思想是在遍历数组的过程中,假设在序号 i 之前的元素即 [0..i-1] 都已经排好序,本趟需要找到 i 对应的元素 x 的正确位置 k ,并且在寻找这个位置 k 的过程中逐个将比较过的元素往后移一位,为元素 x “腾位置”,最后将 k 对应的元素值赋为 x ,插入排序也是根据排序的特性来命名的。

以下是一个实例,红色 标记的数字为插入的数字,被划掉的数字是未参与此次排序的元素,红色 标记的数字与被划掉数字之间的元素为逐个向后移动的元素,比如第二趟参与排序的元素为 [11, 31, 12] ,需要插入的元素为 12 ,但是 12 当前并没有处于正确的位置,于是我们需要依次与前面的元素 31 11 做比较,一边比较一边移动比较过的元素,直到找到第一个比 12 小的元素 11 时停止比较,此时 31 对应的索引 1 则是 12 需要插入的位置。

初始:    [11, 31, 12, 5, 34, 30, 26, 38, 36, 18]

第一趟: [11, 31 , 12, 5, 34, 30, 26, 38, 36, 18] (无移动的元素)

第二趟: [11, 12 , 31, 5, 34, 30, 26, 38, 36, 18] 31 向后移动)

第三趟: [5 , 11, 12, 31, 34, 30, 26, 38, 36, 18] 11, 12, 31 皆向后移动)

第四趟: [5, 11, 12, 31, 34 , 30, 26, 38, 36, 18] (无移动的元素)

第五趟: [5, 11, 12, 30 , 31, 34, 26, 38, 36, 18] 31, 34 向后移动)

第六趟: [5, 11, 12, 26 , 30, 31, 34, 38, 36, 18] 30, 31, 34 向后移动)

第七趟: [5, 11, 12, 26, 30, 31, 34, 38 , 36, 18] (无移动的元素)

第八趟: [5, 11, 12, 26, 30, 31, 34, 36 , 38, 18] 38 向后移动)

第九趟: [5, 11, 12, 18 , 26, 30, 31, 34, 36, 38] 26, 30, 31, 34, 36, 38 向后移动)

插入排序会优于选择排序,理由是它在排序过程中能够利用前部分数组元素已经排好序的一个优势,有效地减少一些比较的次数,当然这种优势得看数组的初始顺序如何,最坏的情况下(给定的数组恰好为倒序)插入排序需要比较和移动的次数将会等于 1 + 2 + 3… + n = n * (n + 1) / 2 ,这种极端情况下,插入排序的效率甚至比选择排序更差。因此插入排序是一个不稳定的排序方法,插入效率与数组初始顺序息息相关。一般情况下,插入排序的时间复杂度和空间复杂度分别为 O(n2 ) O(1)

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Insertion Sorting  
  3.  */  
  4. INSERTION(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         int len = array.length;   
  7.         for (int i = 1; i < len; i++) {   
  8.             T toInsert = array[i];   
  9.             int j = i;   
  10.             for (; j > 0; j--) {   
  11.                 int compare = array[j - 1].compareTo(toInsert);   
  12.                 if (compare == 0 || compare < 0 == ascend) {   
  13.                     break;   
  14.                 }   
  15.                 array[j] = array[j - 1];   
  16.             }   
  17.   
  18.             array[j] = toInsert;   
  19.         }   
  20.     }   
  21. })  
	/**
	 * Insertion Sorting
	 */
	INSERTION(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			int len = array.length;
			for (int i = 1; i < len; i++) {
				T toInsert = array[i];
				int j = i;
				for (; j > 0; j--) {
					int compare = array[j - 1].compareTo(toInsert);
					if (compare == 0 || compare < 0 == ascend) {
						break;
					}
					array[j] = array[j - 1];
				}

				array[j] = toInsert;
			}
		}
	})

 

3. 冒泡排序

冒泡排序可以算是最经典的排序算法了,记得小弟上学时最先接触的也就是这个算法了,因为实现方法最简单,两层 for 循环,里层循环中判断相邻两个元素是否逆序,是的话将两个元素交换,外层循环一次,就能将数组中剩下的元素中最小的元素“浮”到最前面,所以称之为冒泡排序。

照例举个简单的实例吧:

初始状态:   [24, 19, 26, 39, 36, 7, 31, 29, 38, 23]

内层第一趟: [24, 19, 26, 39, 36, 7, 31, 29, 23 , 38 ] 9th [23]<->8th [38

内层第二趟: [24, 19, 26, 39, 36, 7, 31, 23 , 29 , 38] 8th [23]<->7th [29]

内层第三趟: [24, 19, 26, 39, 36, 7, 23 , 31 , 29, 38] 7th [23]<->6th [31]

内层第四趟: [24, 19, 26, 39, 36, 7, 23, 31, 29, 38] 7 23 都位于正确的顺序,无需交换)

内层第五趟: [24, 19, 26, 39, 7 , 36 , 23, 31, 29, 38] 5th [7]<->4th [36]

内层第六趟: [24, 19, 26, 7 , 39 , 36, 23, 31, 29, 38] 4th [7]<->3rd [39]

内层第七趟: [24, 19, 7 , 26 , 39, 36, 23, 31, 29, 38] 3rd [7]<->2nd [26]

内层第八趟: [24, 7 , 19 , 26, 39, 36, 23, 31, 29, 38] 2nd [7]<->1st [19]

内层第九趟: [7 , 24 , 19, 26, 39, 36, 23, 31, 29, 38] 1st [7]<->0th [24]

…...

其实冒泡排序跟选择排序比较相像,比较次数一样,都为 n * (n + 1) / 2 ,但是冒泡排序在挑选最小值的过程中会进行额外的交换(冒泡排序在排序中只要发现相邻元素的顺序不对就会进行交换,与之对应的是选择排序,只会在内层循环比较结束之后根据情况决定是否进行交换),所以在我看来,选择排序属于冒泡排序的改进版。

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Bubble Sorting, it's very similar with Insertion Sorting  
  3.  */  
  4. BUBBLE(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         int length = array.length;   
  7.         int lastExchangedIdx = 0;   
  8.         for (int i = 0; i < length; i++) {   
  9.             // mark the flag to identity whether exchange happened to false   
  10.             boolean isExchanged = false;   
  11.             // last compare and exchange happened before reaching index i   
  12.             int currOrderedIdx = lastExchangedIdx > i ? lastExchangedIdx : i;   
  13.             for (int j = length - 1; j > currOrderedIdx; j--) {   
  14.                 int compare = array[j - 1].compareTo(array[j]);   
  15.                 if (compare != 0 && compare > 0 == ascend) {   
  16.                     exchange(array, j - 1, j);   
  17.                     isExchanged = true;   
  18.                     lastExchangedIdx = j;   
  19.                 }   
  20.             }   
  21.             // if no exchange happen means array is already in order   
  22.             if (isExchanged == false) {   
  23.                 break;   
  24.             }   
  25.         }   
  26.     }   
  27. })  
	/**
	 * Bubble Sorting, it's very similar with Insertion Sorting
	 */
	BUBBLE(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			int length = array.length;
			int lastExchangedIdx = 0;
			for (int i = 0; i < length; i++) {
				// mark the flag to identity whether exchange happened to false
				boolean isExchanged = false;
				// last compare and exchange happened before reaching index i
				int currOrderedIdx = lastExchangedIdx > i ? lastExchangedIdx : i;
				for (int j = length - 1; j > currOrderedIdx; j--) {
					int compare = array[j - 1].compareTo(array[j]);
					if (compare != 0 && compare > 0 == ascend) {
						exchange(array, j - 1, j);
						isExchanged = true;
						lastExchangedIdx = j;
					}
				}
				// if no exchange happen means array is already in order
				if (isExchanged == false) {
					break;
				}
			}
		}
	})

 

 

4. 希尔排序

希尔排序的诞生是由于插入排序在处理大规模数组的时候会遇到需要移动太多元素的问题。希尔排序的思想是将一个大的数组“分而治之”,划分为若干个小的数组,以 gap 来划分,比如数组 [1, 2, 3, 4, 5, 6, 7, 8] ,如果以 gap = 2 来划分,可以分为 [1, 3, 5, 7] [2, 4, 6, 8] 两个数组(对应的,如 gap = 3 ,则划分的数组为: [1, 4, 7] [2, 5, 8] [3, 6] )然后分别对划分出来的数组进行插入排序,待各个子数组排序完毕之后再减小 gap 值重复进行之前的步骤,直至 gap = 1 ,即对整个数组进行插入排序,此时的数组已经基本上快排好序了,所以需要移动的元素会很小很小,解决了插入排序在处理大规模数组时较多移动次数的问题。

具体实例请参照插入排序。

希尔排序是插入排序的改进版,在数据量大的时候对效率的提升帮助很大,数据量小的时候建议直接使用插入排序就好了。

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Shell Sorting  
  3.  */  
  4. SHELL(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         int length = array.length;   
  7.         int gap = 1;   
  8.   
  9.         // use the most next to length / 3 as the first gap   
  10.         while (gap < length / 3) {   
  11.             gap = gap * 3 + 1;   
  12.         }   
  13.   
  14.         while (gap >= 1) {   
  15.             for (int i = gap; i < length; i++) {   
  16.                 T next = array[i];   
  17.                 int j = i;   
  18.                 while (j >= gap) {   
  19.                     int compare = array[j - gap].compareTo(next);   
  20.                     // already find its position   
  21.                     if (compare == 0 || compare < 0 == ascend) {   
  22.                         break;   
  23.                     }   
  24.   
  25.                     array[j] = array[j - gap];   
  26.                     j -= gap;   
  27.                 }   
  28.                 if (j != i) {   
  29.                     array[j] = next;   
  30.                 }   
  31.             }   
  32.             gap /= 3;   
  33.         }   
  34.   
  35.     }   
  36. })  
	/**
	 * Shell Sorting
	 */
	SHELL(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			int length = array.length;
			int gap = 1;

			// use the most next to length / 3 as the first gap
			while (gap < length / 3) {
				gap = gap * 3 + 1;
			}

			while (gap >= 1) {
				for (int i = gap; i < length; i++) {
					T next = array[i];
					int j = i;
					while (j >= gap) {
						int compare = array[j - gap].compareTo(next);
						// already find its position
						if (compare == 0 || compare < 0 == ascend) {
							break;
						}

						array[j] = array[j - gap];
						j -= gap;
					}
					if (j != i) {
						array[j] = next;
					}
				}
				gap /= 3;
			}

		}
	})

 

 

5. 归并排序

归并排序采用的是递归来实现,属于“分而治之”,将目标数组从中间一分为二,之后分别对这两个数组进行排序,排序完毕之后再将排好序的两个数组“归并”到一起,归并排序最重要的也就是这个“归并”的过程,归并的过程中需要额外的跟需要归并的两个数组长度一致的空间,比如需要规定的数组分别为: [3, 6, 8, 11] [1, 3, 12, 15] (虽然逻辑上被划为为两个数组,但实际上这些元素还是位于原来数组中的,只是通过一些 index 将其划分成两个数组,原数组为 [3, 6, 8, 11, 1, 3, 12, 15 ,我们设置三个指针 lo, mid, high 分别为 0,3,7 就可以实现逻辑上的子数组划分)那么需要的额外数组的长度为 4 + 4 = 8 。归并的过程可以简要地概括为如下:

1) 将两个子数组中的元素复制到新数组 copiedArray 中,以前面提到的例子为例,则 copiedArray = [3, 6, 8, 11, 1, 3, 12, 15]

2) 设置两个指针分别指向原子数组中对应的第一个元素,假定这两个指针取名为 leftIdx rightIdx ,则 leftIdx = 0 (对应 copiedArray 中的第一个元素 [3] ), rightIdx = 4 (对应 copiedArray 中的第五个元素 [1] );

3) 比较 leftIdx rightIdx 指向的数组元素值,选取其中较小的一个并将其值赋给原数组中对应的位置 i ,赋值完毕后分别对参与赋值的这两个索引做自增 1 操作,如果 leftIdx rigthIdx 值已经达到对应数组的末尾,则余下只需要将剩下数组的元素按顺序 copy 到余下的位置即可。

下面给个归并的具体实例:

第一趟:

辅助数组 [21 , 28, 39 | 35, 38] (数组被拆分为左右两个子数组,以 | 分隔开)

[21 ,  ,  ,  ,  ] (第一次 21 35 比较 , 左边子数组胜出, leftIdx = 0 i = 0

第二趟:

辅助数组 [21, 28 , 39 | 35, 38]

[21 , 28,  ,  ,  ] (第二次 28 35 比较,左边子数组胜出, leftIdx = 1 i = 1

第三趟: [21, 28, 39 | 35 , 38]

 [21 , 28 , 35,  ,  ] (第三次 39 35 比较,右边子数组胜出, rightIdx = 0 i = 2

第四趟: [21, 28, 39 | 35, 38 ]

 [21 , 28 , 35 , 38,  ] (第四次 39 38 比较,右边子数组胜出, rightIdx = 1 i = 3

第五趟: [21, 28, 39 | 35, 38]

 [21 , 28 , 35 , 38 , 39] (第五次时右边子数组已复制完,无需比较 leftIdx = 2 i = 4

以上便是一次归并的过程,我们可以将整个需要排序的数组做有限次拆分(每次一分为二)直到分为长度为 1 的小数组为止,长度为 1 时数组已经不用排序了。在这之后再逆序(由于采用递归)依次对这些数组进行归并操作,直到最后一次归并长度为 n / 2 的子数组,归并完成之后数组排序也完成。

归并排序需要的额外空间是所有排序中最多的,每次归并需要与参与归并的两个数组长度之和相同个元素(为了提供辅助数组)。则可以推断归并排序的空间复杂度为 1 + 2 + 4 + … + n = n * ( n + 2) / 4 (忽略了 n 的奇偶性的判断),时间复杂度比较难估,这里小弟也忘记是多少了(囧)。

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Merge sorting  
  3.  */  
  4. MERGE(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         this.sort(array, 0, array.length - 1, ascend);   
  7.     }   
  8.   
  9.     private <T extends Comparable<T>> void sort(T[] array, int lo, int hi, boolean ascend) {   
  10.         // OPTIMIZE ONE   
  11.         // if the substring's length is less than 20,   
  12.         // use insertion sort to reduce recursive invocation   
  13.         if (hi - lo < 20) {   
  14.             for (int i = lo + 1; i <= hi; i++) {   
  15.                 T toInsert = array[i];   
  16.                 int j = i;   
  17.                 for (; j > lo; j--) {   
  18.                     int compare = array[j - 1].compareTo(toInsert);   
  19.                     if (compare == 0 || compare < 0 == ascend) {   
  20.                         break;   
  21.                     }   
  22.                     array[j] = array[j - 1];   
  23.                 }   
  24.   
  25.                 array[j] = toInsert;   
  26.             }   
  27.   
  28.             return;   
  29.         }   
  30.   
  31.         int mid = lo + (hi - lo) / 2;   
  32.         sort(array, lo, mid, ascend);   
  33.         sort(array, mid + 1, hi, ascend);   
  34.         merge(array, lo, mid, hi, ascend);   
  35.     }   
  36.   
  37.     private <T extends Comparable<T>> void merge(T[] array, int lo, int mid, int hi, boolean ascend) {   
  38.         // OPTIMIZE TWO   
  39.         // if it is already in right order, skip this merge   
  40.         // since there's no need to do so   
  41.         int leftEndCompareToRigthStart = array[mid].compareTo(array[mid + 1]);   
  42.         if (leftEndCompareToRigthStart == 0 || leftEndCompareToRigthStart < 0 == ascend) {   
  43.             return;   
  44.         }   
  45.   
  46.         @SuppressWarnings("unchecked")   
  47.         T[] arrayCopy = (T[]) new Comparable[hi - lo + 1];   
  48.         System.arraycopy(array, lo, arrayCopy, 0, arrayCopy.length);   
  49.   
  50.         int lowIdx = 0;   
  51.         int highIdx = mid - lo + 1;   
  52.   
  53.         for (int i = lo; i <= hi; i++) {   
  54.             if (lowIdx > mid - lo) {   
  55.                 // left sub array exhausted   
  56.                 array[i] = arrayCopy[highIdx++];   
  57.             } else if (highIdx > hi - lo) {   
  58.                 // right sub array exhausted   
  59.                 array[i] = arrayCopy[lowIdx++];   
  60.             } else if (arrayCopy[lowIdx].compareTo(arrayCopy[highIdx]) < 0 == ascend) {   
  61.                 array[i] = arrayCopy[lowIdx++];   
  62.             } else {   
  63.                 array[i] = arrayCopy[highIdx++];   
  64.             }   
  65.         }   
  66.     }   
  67. })  
	/**
	 * Merge sorting
	 */
	MERGE(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			this.sort(array, 0, array.length - 1, ascend);
		}

		private <T extends Comparable<T>> void sort(T[] array, int lo, int hi, boolean ascend) {
			// OPTIMIZE ONE
			// if the substring's length is less than 20,
			// use insertion sort to reduce recursive invocation
			if (hi - lo < 20) {
				for (int i = lo + 1; i <= hi; i++) {
					T toInsert = array[i];
					int j = i;
					for (; j > lo; j--) {
						int compare = array[j - 1].compareTo(toInsert);
						if (compare == 0 || compare < 0 == ascend) {
							break;
						}
						array[j] = array[j - 1];
					}

					array[j] = toInsert;
				}

				return;
			}

			int mid = lo + (hi - lo) / 2;
			sort(array, lo, mid, ascend);
			sort(array, mid + 1, hi, ascend);
			merge(array, lo, mid, hi, ascend);
		}

		private <T extends Comparable<T>> void merge(T[] array, int lo, int mid, int hi, boolean ascend) {
			// OPTIMIZE TWO
			// if it is already in right order, skip this merge
			// since there's no need to do so
			int leftEndCompareToRigthStart = array[mid].compareTo(array[mid + 1]);
			if (leftEndCompareToRigthStart == 0 || leftEndCompareToRigthStart < 0 == ascend) {
				return;
			}

			@SuppressWarnings("unchecked")
			T[] arrayCopy = (T[]) new Comparable[hi - lo + 1];
			System.arraycopy(array, lo, arrayCopy, 0, arrayCopy.length);

			int lowIdx = 0;
			int highIdx = mid - lo + 1;

			for (int i = lo; i <= hi; i++) {
				if (lowIdx > mid - lo) {
					// left sub array exhausted
					array[i] = arrayCopy[highIdx++];
				} else if (highIdx > hi - lo) {
					// right sub array exhausted
					array[i] = arrayCopy[lowIdx++];
				} else if (arrayCopy[lowIdx].compareTo(arrayCopy[highIdx]) < 0 == ascend) {
					array[i] = arrayCopy[lowIdx++];
				} else {
					array[i] = arrayCopy[highIdx++];
				}
			}
		}
	})

 

6. 快速排序

快速排序也是用归并方法实现的一个“分而治之”的排序算法,它的魅力之处在于它能在每次partition(排序算法的核心所在)都能为一个数组元素确定其排序最终正确位置(一次就定位准,下次循环就不考虑这个元素了)。

快速排序的partition操作按以下逻辑进行,假定本次排序的数组为arr

1) 选择一个元素(为了简单起见,就选择本次partition的第一个元素,即arr[0])作为基准元素,接下来的步骤会为其确定排序完成后最终的位置;

2) 1)  接下来需要遍历[1…n-1]对应的数组元素以帮助找到arr[0]值(以v替代)对应的位置,定义i为当前访问数组的索引,lt为值小于v的最大索引,gt为值大于v的最小索引,那么在遍历过程中,如果发现i指向的值与v相等,则将i值加1,继续下一次比较;如果i指向的值比v小,则将ilt对应的元素进行交换,然后分别将两个索引加1;如果i指向的值比v大,则将igt对应的元素进行交换,然后i自增,gt自减。循环遍历完成(i > gt时结束)之后可以保证[0…lt-1]对应的值都是比v小的,[lt..gt]之间的值都是与v相等的,[gt+1…n-1]对应的值都是比v大的。

3) 分别对[0…lt-1][gt+1…n-1]两个子数组进行排序,如此递归,直至子子子数组的长度为0

 

下面举个partition的具体实例:

初始(i = 1, lt = 0, gt = 8):

       [41, 59, 43, 26, 63, 30, 29, 26, 42](需要确定位置的为0th[41]

第一趟(i = 1, lt = 0, gt = 8):

       [41, 42, 43, 26, 63, 30, 29, 26, 59]1st[59] > 411st[59]<->8th[42]gt--

第二趟(i = 1, lt = 0, gt = 7):

       [41, 26, 43, 26, 63, 30, 29, 42, 59]1st[42] > 411st[42]<->7th[26]gt--

第三趟(i = 1, lt = 0, gt = 6):

       [26, 41, 43, 26, 63, 30, 29, 42, 59]1st[26] < 41, 1st[26]<->0st[41]i++, lt++

第四趟(i = 2, lt = 1, gt = 6):

       [26, 41, 29, 26, 63, 30, 43, 42, 59]2nd[43] > 412nd[43]<->6th[29]gt--

第五趟(i = 2, lt = 1, gt = 5):

       [26, 29, 41, 26, 63, 30, 43, 42, 59]2nd[29] < 41, 2nd[29]<->1st[41]i++lt++

第六趟(i = 3, lt = 2, gt = 5):    

       [26, 29, 26, 41, 63, 30, 43, 42, 59]3rd[26] < 413rd[26]<->2nd[41]i++lt++

第七趟(i = 4, lt = 3, gt = 5):

       [26, 29, 26, 41, 30, 63, 43, 42, 59] 4th[63] > 414th[63]<->5th[30]gt--

第八趟(i = 4, lt = 3, gt = 4):    

       [26, 29, 26, 30, 41, 63, 43, 42, 59]4th[30] < 414th[30]<->3rd[41]i++lt++

 

可以看出,在一次partition之后,以41为分割线,41左侧皆为比它小的元素,41右侧皆为比它大或相等的元素(当然这个实例比较特殊,没有出现和41相等的元素)。快速排序顾名思义就是排序速度非常快,后面我会放在我机器上跑各个排序方法的时间对比图。值得一提的是JDK中在Arrays工具内中内置的sort方法就是接合插入排序和三路快速排序实现的,有兴趣的同学可以看看JDK的源码。

 

实现代码:

 

Java代码 复制代码 收藏代码
  1. /**  
  2.  * Quick Sorting  
  3.  */  
  4. QUICK(new Sortable() {   
  5.     public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {   
  6.         this.sort(array, 0, array.length - 1, ascend);   
  7.     }   
  8.   
  9.     private <T extends Comparable<T>> void sort(T[] array, int lo, int hi, boolean ascend) {   
  10.         if (lo >= hi) {   
  11.             return;   
  12.         }   
  13.   
  14.         T toFinal = array[lo];   
  15.         int leftIdx = lo;   
  16.         int rightIdx = hi;   
  17.   
  18.         int i = lo + 1;   
  19.   
  20.         while (i <= rightIdx) {   
  21.             int compare = array[i].compareTo(toFinal);   
  22.             if (compare == 0) {   
  23.                 i++;   
  24.             } else if (compare < 0 == ascend) {   
  25.                 exchange(array, leftIdx++, i++);   
  26.             } else {   
  27.                 exchange(array, rightIdx--, i);   
  28.             }   
  29.         }   
  30.   
  31.         // partially sort left array and right array   
  32.         // no need to include the leftIdx-th to rightIdx-th elements   
  33.         // since they are already in its final position   
  34.         sort(array, lo, leftIdx - 1, ascend);   
  35.         sort(array, rightIdx + 1, hi, ascend);   
  36.     }   
  37. })  
	/**
	 * Quick Sorting
	 */
	QUICK(new Sortable() {
		public <T extends Comparable<T>> void sort(T[] array, boolean ascend) {
			this.sort(array, 0, array.length - 1, ascend);
		}

		private <T extends Comparable<T>> void sort(T[] array, int lo, int hi, boolean ascend) {
			if (lo >= hi) {
				return;
			}

			T toFinal = array[lo];
			int leftIdx = lo;
			int rightIdx = hi;

			int i = lo + 1;

			while (i <= rightIdx) {
				int compare = array[i].compareTo(toFinal);
				if (compare == 0) {
					i++;
				} else if (compare < 0 == ascend) {
					exchange(array, leftIdx++, i++);
				} else {
					exchange(array, rightIdx--, i);
				}
			}

			// partially sort left array and right array
			// no need to include the leftIdx-th to rightIdx-th elements
			// since they are already in its final position
			sort(array, lo, leftIdx - 1, ascend);
			sort(array, rightIdx + 1, hi, ascend);
		}
	})

 

 

如果你希望查看完整代码,请移驾至我的GoogleCode查看,传送门

这里是我测试时的测试用例,利用了枚举和策略模式的优势,切换排序算法时相对会比较容易。

 

以下为经典排序算法在我机器上运行的耗时对比图(测试用的随机数组长度为50000),直接截的测试用例的图。

 

 

转载: http://easense2009.iteye.com/blog/1568614

分享到:
评论

相关推荐

    常用排序算法Java实现

    这里我们主要关注Java实现的七大经典排序算法:冒泡排序、插入排序、选择排序、希尔排序、快速排序、归并排序以及堆排序。 1. 冒泡排序(Bubble Sort): 冒泡排序是最简单的排序方法之一,它通过重复遍历待排序的...

    常用排序算法java演示

    本文将深入探讨标题"常用排序算法java演示"中涉及的知识点,包括排序算法的原理、Java实现方式以及其在实际应用中的图形演示。 首先,让我们逐一了解几种常见的排序算法: 1. **冒泡排序(Bubble Sort)**:这是一...

    常用排序算法的java实现(冒泡、插入、选择、希尔、归并、快排)

    本篇文章将详细讲解标题中提到的六种常见排序算法的Java实现。 1. **冒泡排序**:冒泡排序是最基础的排序算法之一,它通过不断交换相邻的逆序元素来逐渐将较大的元素“浮”到数组的前端。在Java中,冒泡排序的基本...

    常用各种排序算法Java的实现_差不多了__.rar

    本资源"常用各种排序算法Java的实现_差不多了__.rar"显然是一个包含了各种经典排序算法Java实现的压缩文件,对于学习和理解这些算法的开发者来说极具价值。 首先,我们来概述一下常见的排序算法: 1. 冒泡排序:是...

    常用排序算法Java

    以上就是Java中实现的一些常用排序算法,它们各有优缺点,适用于不同的场景。理解并熟练掌握这些排序算法,有助于优化代码性能,提高编程能力。在实际开发中,应根据具体需求选择合适的排序算法,以达到最佳的效率和...

    常用的排序算法(java实现),附带一个PPT动画演示、详解了其中三种

    这里我们主要关注Java实现的排序算法,并结合一个PPT的动画演示来探讨其中的插入排序、直接插入排序和希尔排序。 首先,让我们深入理解插入排序。插入排序是一种简单的排序算法,其基本思想是将未排序的元素逐个...

    Java实现常用排序算法

    虽然树形选择排序和堆排序在这次实现中未涵盖,但理解这四种排序算法的基本原理和Java实现方式对于提升编程技能至关重要。 首先,让我们来看看插入排序。插入排序是一种简单直观的排序算法,它的工作原理类似于人们...

    常用排序算法分析与实现(Java版)

    ### 常用排序算法分析与实现(Java版) #### 插入排序 **1. 直接插入排序** 直接插入排序是一种简单的排序方法,它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并...

    常用排序算法JAVA版

    这个名为"常用排序算法JAVA版"的压缩包文件很可能包含了Java实现的各种经典排序算法,如冒泡排序、选择排序、插入排序、快速排序、归并排序、堆排序等。 1. **冒泡排序**:是最简单的排序算法之一,通过不断交换...

    JAVA写的6种内部排序算法简单实现

    以上就是六种常用的内部排序算法在Java语言中的实现。理解这些排序算法的原理和性能特点,有助于在实际编程中选择合适的排序方法,提高程序效率。对于面试或者笔试,熟练掌握这些算法将大大提高你的竞争力。在实践中...

    常用排序算法总结(含Java代码)

    冒泡排序和快速排序是两种基础但广泛使用的数据排序算法。冒泡排序由于其简单直观的特性,易于理解和实现,而快速排序则以其较高的效率在数据量较大时展现出优势。 首先,让我们来看冒泡排序算法。冒泡排序通过重复...

    常用排序算法源码下载(Java实现)

    常用排序算法的Java实现源码,包括冒泡排序,快速排序,直接插入排序,希尔排序,直接选择排序,堆排序,归并排序,基数排序,计数排序。

    八大排序算法总结(含Java实现源代码)

    在计算机科学中,排序算法是数据处理的重要组成部分,它们用于将一组无序的数据按照特定顺序进行排列。这里我们将深入探讨八大排序算法,并结合Java语言来理解它们的实现原理。 1. 冒泡排序(Bubble Sort) 冒泡...

    Java常用排序算法源码

    在编程领域,排序算法是计算机科学中的核心概念,特别是在Java这样的高级编程语言中。排序算法是用来组织和优化数据结构的关键工具,使得数据按照特定规则(如升序或降序)排列。以下是对Java中几种常见排序算法的...

    常用排序算法实现(java)

    本资源提供了五种经典的排序算法的Java实现,包括选择排序(Selection Sort)、插入排序(Insertion Sort)、希尔排序(Shell Sort)、归并排序(Merge Sort)以及快速排序(Quick Sort)。 1. **选择排序**:选择排序是一种...

    Java常用排序算法程序员必须掌握的8大排序算法Java开

    学习资料如"Java常用排序算法程序员必须掌握的8大排序算法Java开发Java经验技巧共16页.pdf"可以提供详细的讲解和示例,帮助你更好地理解和实践这些算法。同时,这些排序算法不仅限于Java,也广泛应用于Python、C语言...

    Java常用8大排序算法

    ### Java常用八大排序算法详解 #### 一、直接插入排序 **基本思想:** 直接插入排序的基本思路是在要排序的一组数中,假设前面 (n-1) [n&gt;=2] 个数已经排好顺序,现在要把第 n 个数插入到前面的有序数列中,使得这 ...

    常用排序算法小结(附Java实现)

    这篇博客“常用排序算法小结(附Java实现)”提供了一种深入理解并掌握常见排序算法的途径,尤其对于Java开发者来说非常实用。文章可能涵盖了如冒泡排序、选择排序、插入排序、快速排序、归并排序、堆排序等多种经典...

    Java常用排序算法&程序员必须掌握的8大排序算法+二分法查找

    本资料包聚焦于"Java常用排序算法"和"程序员必须掌握的8大排序算法",并深入探讨了"二分法查找"这一高效搜索技术。 首先,我们来看八大排序算法。这些算法包括: 1. **冒泡排序**:最简单的排序方法,通过不断交换...

Global site tag (gtag.js) - Google Analytics