- [1 å½å¹¶æåºãéæºå¿«æ](#1)
* [1.1 å½å¹¶æåº](#11)
+ [1.1.1 éå½æè·¯ï¼](#111)
+ [1.1.2 ééå½æè·¯](#112)
+ [1.1.3 å½å¹¶æåºæ¶é´å¤æåº¦](#113)
+ [1.1.4 å½å¹¶é¢è¯é¢](#114)
* [1.2 å¿«æ](#12)
+ [1.2.1 Partionè¿ç¨](#121)
+ [1.2.2 å¿«æ1.0ï¼æ¯æ¬¡partionæå®ä¸ä¸ªä½ç½®](#122)
+ [1.2.3 å¿«æ2.0ï¼æ¯æ¬¡partionæå®ä¸æ¹ä½ç½®](#123)
+ [1.2.4 å¿«æ3.0ï¼éæºä½ç½®ä½ä¸ºnumæ è®°ä½](#124)
+ [1.2.5 å¿«æçæ¶é´å¤æåº¦ä¸ç©ºé´å¤æåº¦](#125)
1 å½å¹¶æåºãéæºå¿«æ
>
1.1 å½å¹¶æåº
>
1ã æ´ä½æ¯éå½çï¼å·¦è¾¹æå¥½åºå³è¾¹æå¥½åºï¼æåmerge让æ´ä½æåºï¼mergeè¿ç¨éè¦ç³è¯·å被æåºæ°ç»çé¿åº¦çè¾
å©ç©ºé´
2ã 让å
¶æ´ä½æåºçè¿ç¨éç¨äºæå¤åºçæ¹æ³
3ã å©ç¨masterå
¬å¼æ¥æ±è§£å½å¹¶çæ¶é´å¤æåº¦
4ã å½å¹¶æåºå¯æ¹ä¸ºééå½å®ç°
1.1.1 éå½æè·¯ï¼
>
> 䏻彿°å¸æä¸ä¸ªæ°ç»ç0~3ä½ç½®æåºf(arr, 0, 3)
> 第ä¸å±éå½å¸æf(arr, 0, 1)åf(arr, 2, 3)å嫿åºã
> 第äºå±éå½ï¼f(arr, 0, 1)叿f(arr, 0, 0)åf(arr, 1, 1)æåº, f(arr, 2, 3)叿f(arr, 2, 2)åf(arr, 3, 3)å嫿åºã
> f(arr, 0, 0)åf(arr, 1, 1)å·²ç»æåºï¼åå°ç¬¬ä¸å±éå½f(arr, 0, 1)ä¸å»merge0ä½ç½®çæ°å1ä½ç½®çæ°åå·åå
ç´ ç»ç0å°1ä½ç½®ï¼0å°1ä½ç½®å为æåº; f(arr, 2, 2)åf(arr, 3, 3)å·²ç»æåºï¼åå°f(arr, 2, 3)ä¸å»merge2ä½ç½®çæ°å3ä½ç½®çæ°åå·ååæ°ç»ç2å°3ä½ç½®ï¼2å°3ä½ç½®å为æåºã
> f(arr, 0, 3)éè¦merge f(arr, 0, 1)åf(arr, 2, 3)æ¤æ¶f(arr, 0, 1)åf(arr, 2, 3)å·²ç»æåºmergeåcopyå°åæ°ç»ç0å°3ä½ç½®ãäºæ¯f(arr, 0, 3)æ´ä½æåº
1.1.2 ééå½æè·¯
>
> 对äºä¸ä¸ªç»å®é¿åº¦ä¸ºnçæ°ç»arrï¼æä»¬å¸æarræåº
> åå§åç»ä¸ºa=2ï¼æä»¬è®©æ¯ä¸¤ä¸ªæåºï¼ä¸å¤ä¸ç»ç彿ä¸ç»
> åç»å为a=2*2=4,ç±äºä¸ä¸æ¥å·²ç»ä¿è¯äºä¸¤ä¸¤æåºï¼é£ä¹æä»¬å¯ä»¥å½ååç»çå个æ°çå两个åå两个æ°mergeä½¿å¾æ¯åä¸ªæ°æåº
> åç»å为a=2*4=8ï¼...ç´è³a>=n,æ´ä½æåº
```Java
package class03;
public class Code01_MergeSort {
// é彿¹æ³å®ç°
public static void mergeSort1(int[] arr) {
if (arr == null || arr.length < 2) {
return;
}
// ä¼ å
¥è¢«æåºæ°ç»ï¼ä»¥åå·¦å³è¾¹ç
process(arr, 0, arr.length - 1);
}
// arr[L...R]èå´ä¸ï¼åææåºç
// L...R N T(N) = 2*T(N/2) + O(N) ->
public static void process(int[] arr, int L, int R) {
if (L == R) { // base case
return;
}
// >> æç¬¦å·å³ç§»1ä½ï¼ç¸å½äºé¤ä»¥2
int mid = L + ((R - L) >> 1);
process(arr, L, mid);
process(arr, mid + 1, R);
// å½åæ é¡¶å·¦å³å·²ç»æå¥½åºï¼åå¤å·¦å³mergeï¼æ³¨æè¿éçmergeéå½çæ¯ä¸å±é½ä¼è°ç¨
merge(arr, L, mid, R);
}
public static void merge(int[] arr, int L, int M, int R) {
// mergeè¿ç¨ç³è¯·è¾
婿°ç»ï¼åå¤copy
int[] help = new int[R - L + 1];
// ç¨æ¥æ è¯helpç䏿
int i = 0;
// 左边æåºæ°ç»çæé
int p1 = L;
// å³è¾¹æåºæ°ç»çæé
int p2 = M + 1;
// p1åp2齿²¡è¶ççæ
åµä¸ï¼è°å°copyè°
while (p1 <= M && p2 <= R) {
help[i++] = arr[p1] <= arr[p2] ? arr[p1++] : arr[p2++];
}
// è¦ä¹p1è¶çäºï¼è¦ä¹p2è¶çäºï¼è°æ²¡è¶çæè°å©ä¸çå
ç´ copyå°helpä¸
while (p1 <= M) {
help[i++] = arr[p1++];
}
while (p2 <= R) {
help[i++] = arr[p2++];
}
// æè¾
婿°ç»ä¸æ´ä½mergeåçæåºæ°ç»ï¼copyååæ°ç»ä¸å»
for (i = 0; i < help.length; i++) {
arr[L + i] = help[i];
}
}
// éé彿¹æ³å®ç°
public static void mergeSort2(int[] arr) {
if (arr == null || arr.length < 2) {
return;
}
int N = arr.length;
// å½åæåºçï¼å·¦ç»é¿åº¦,é£ä¹å®è´¨åç»å¤§å°æ¯ä»2å¼å§ç
int mergeSize = 1;
while (mergeSize < N) { // log N
// L表示å½ååç»çå·¦ç»çä½ç½®ï¼åå§ä¸ºç¬¬ä¸ä¸ªåç»çå·¦ç»ä½ç½®ä¸º0
int L = 0;
// 0....
while (L < N) {
// L...M å½åå·¦ç»ï¼mergeSizeï¼
int M = L + mergeSize - 1;
// å½åå·¦ç»å
å«å½ååç»çææå
ç´ ï¼å³æ²¡æå³ç»äºï¼æ émergeå·²ç»æåº
if (M >= N) {
break;
}
// L...Mä¸ºå·¦ç» M+1...R(mergeSize)为å³ç»ãå³ç»å¤mergeSizeä¸ªçæ¶åï¼å³åæ 为M + mergeSizeï¼å³ç»ä¸å¤çæ
åµä¸å³ç»è¾¹çåæ ä¸ºæ´ä¸ªæ°ç»å³è¾¹çN - 1
int R = Math.min(M + mergeSize, N - 1);
// æå½åç»è¿è¡merge
merge(arr, L, M, R);
// ä¸ä¸ä¸ªåç»çå·¦ç»èµ·å§ä½ç½®
L = R + 1;
}
// 妿mergeSizeä¹2å¿
å®å¤§äºNï¼ç´æ¥breakã鲿¢mergeSize溢åºï¼æå¯è½Nå¾å¤§ï¼ä¸é¢ä¹2æå¯è½èå´æº¢åºï¼æ´å½¢æ°å¤§äº21亿ï¼
if (mergeSize > N / 2) {
break;
}
// æ 符å·å·¦ç§»ï¼ç¸å½äºä¹ä»¥2
mergeSize <<= 1;
}
}
// for test
public static int[] generateRandomArray(int maxSize, int maxValue) {
int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
for (int i = 0; i < arr.length; i++) {
arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
}
return arr;
}
// for test
public static int[] copyArray(int[] arr) {
if (arr == null) {
return null;
}
int[] res = new int[arr.length];
for (int i = 0; i < arr.length; i++) {
res[i] = arr[i];
}
return res;
}
// for test
public static boolean isEqual(int[] arr1, int[] arr2) {
if ((arr1 == null && arr2 != null) || (arr1 != null && arr2 == null)) {
return false;
}
if (arr1 == null && arr2 == null) {
return true;
}
if (arr1.length != arr2.length) {
return false;
}
for (int i = 0; i < arr1.length; i++) {
if (arr1[i] != arr2[i]) {
return false;
}
}
return true;
}
// for test
public static void printArray(int[] arr) {
if (arr == null) {
return;
}
for (int i = 0; i < arr.length; i++) {
System.out.print(arr[i] + " ");
}
System.out.println();
}
// for test
public static void main(String[] args) {
int testTime = 500000;
int maxSize = 100;
int maxValue = 100;
boolean succeed = true;
for (int i = 0; i < testTime; i++) {
int[] arr1 = generateRandomArray(maxSize, maxValue);
int[] arr2 = copyArray(arr1);
mergeSort1(arr1);
mergeSort2(arr2);
if (!isEqual(arr1, arr2)) {
succeed = false;
printArray(arr1);
printArray(arr2);
break;
}
}
System.out.println(succeed ? "Nice!" : "Oops!");
}
}
```
1.1.3 å½å¹¶æåºæ¶é´å¤æåº¦
>
> éå½å¤æåº¦è®¡ç®ï¼ç¨masterå
¬å¼å¸¦å
¥ï¼åé®é¢è§æ¨¡N/2ï¼è°ç¨2次ï¼é¤äºéå½ä¹å¤çæ¶é´å¤æåº¦ä¸ºmergeçæ¶é´å¤æåº¦ï¼ä¸ºO(N)ãa=2,b=2,d=1满足masterç¬¬ä¸æ¡logb^a == dè§å
```math
T(N) = 2T(N/2) + O(N) => O(N*logN)
```
> ééå½å¤æåº¦è®¡ç®ï¼mergeSize*2çäºåç»ä»2->4->8->...,æ¯ä¸ªåç»ä¸æ§è¡mergeæä½O(N)ãæä»¥ééå½åéå½çæ¶é´å¤æåº¦ç¸åï¼ä¹ä¸ºO(N)*O(logN) = O(NlogN)
==éå½åééå½çå½å¹¶æåºæ¶é´å¤æåº¦é½ä¸ºO(NlogN)==
Tips: 为ä»ä¹éæ©ï¼åæ³¡ï¼æå
¥æåºçæ¶é´å¤æåº¦ä¸ºO(N^2)èå½å¹¶æåºæ¶é´å¤æåº¦ä¸ºO(NlogN)ï¼å ä¸ºéæ©ï¼åæ³¡ï¼æå
¥æåºçæ¯ä¸ªå
ç´ æµªè´¹äºå¤§éçæ¯è¾è¡ä¸ºN次ãèå½å¹¶æ²¡ææµªè´¹æ¯è¾è¡ä¸ºï¼æ¯æ¬¡æ¯è¾çç»ææåºåé½ä¼ä¿å䏿¥ï¼æç»merge
1.1.4 å½å¹¶é¢è¯é¢
>
1ãå¨ä¸ä¸ªæ°ç»ä¸ï¼ä¸ä¸ªæ°å·¦è¾¹æ¯å®å°çæ°çæ»åï¼å«åå°åï¼æææ°çå°åç´¯å èµ·æ¥ï¼å«åæ°ç»çå°åãæ±æ°ç»çå°åãä¾å¦[1, 3, 4, 2, 5]
```text
1左边æ¯1å°çæ°ï¼æ²¡æ
3左边æ¯3å°çæ°ï¼1
4左边æ¯4å°çæ°ï¼1ã3
2左边æ¯2å°çæ°ä¸ºï¼1
5左边æ¯5å°çæ°ä¸ºï¼1ã3ã4ã2
æä»¥è¯¥æ°ç»çå°å为ï¼1+1+3+1+1+3+4+2 = 16
```
> æ´åè§£æ³ï¼æ¯ä¸ªæ°æ¾ä¹åæ¯èªå·±å°çæ°ï¼ç´¯å èµ·æ¥ï¼æ¶é´å¤æåº¦ä¸ºO(N^2)ï¼é¢è¯æ²¡åã使¯æ´åæ¹æ³å¯ä»¥ç¨æ¥å对æ°å¨
> å½å¹¶æåºè§£æ³æè·¯ï¼O(NlogN)ãå¨éå½mergeçè¿ç¨ä¸ï¼äº§çå°åãè§åæ¯å·¦ç»æ¯å³ç»æ°å°çæ¶å产çå°åï¼é¤æ¤ä¹å¤ä¸äº§çï¼å½å·¦ç»åå³ç»æ°ç¸ççæ¶åï¼æ·è´å³ç»çæ°ï¼ä¸äº§çå°åï¼å½å·¦ç»çæ°å¤§äºå³ç»çæ¶åï¼æ·è´å³ç»çæ°ï¼ä¸äº§çå°åãå®è´¨æ¯ææ¾å·¦è¾¹æ¯æ¬èº«å°çæ°çé®é¢ï¼è½¬å为æ¾è¿ä¸ªæ°å³ä¾§æå¤å°ä¸ªæ°æ¯èªå·±å¤§ï¼å¨æ¯æ¬¡mergeçè¿ç¨ä¸ï¼ä¸ä¸ªæ°å¦æå¤å¨å·¦ç»ä¸ï¼é£ä¹åªä¼å»æ¾å³ç»ä¸æå¤å°ä¸ªæ°æ¯èªå·±å¤§
```Java
package class03;
public class Code02_SmallSum {
public static int smallSum(int[] arr) {
if (arr == null || arr.length < 2) {
return 0;
}
return process(arr, 0, arr.length - 1);
}
// arr[L..R]æ¢è¦æå¥½åºï¼ä¹è¦æ±å°åè¿å
// ææmergeæ¶ï¼äº§ççå°åï¼ç´¯å
// å·¦ æåº merge
// å³ æåº merge
// arr æ´ä½ merge
public static int process(int[] arr, int l, int r) {
// åªæä¸ä¸ªæ°ï¼ä¸åå¨å³ç»ï¼å°å为0
if (l == r) {
return 0;
}
// l < r
int mid = l + ((r - l) >> 1);
// 左侧mergeçå°å+å³ä¾§mergeçå°å+æ´ä½å·¦å³ä¸¤ä¾§çå°å
return process(arr, l, mid) + process(arr, mid + 1, r) + merge(arr, l, mid, r);
}
public static int merge(int[] arr, int L, int m, int r) {
// å¨å½å¹¶æåºçåºç¡ä¸æ¹è¿ï¼å¢å å°åres = 0
int[] help = new int[r - L + 1];
int i = 0;
int p1 = L;
int p2 = m + 1;
int res = 0;
while (p1 <= m && p2 <= r) {
// å½åçæ°æ¯æ¯å³ç»å°çï¼äº§çå³ç»å½åä½ç½®å°å³ç»å³è¾¹çæ°é个å°åï¼ç´¯å å°resãå¦åreså 0
res += arr[p1] < arr[p2] ? (r - p2 + 1) * arr[p1] : 0;
// åªæå·¦ç»å½åæ°å°äºå³ç»copy左边çï¼å¦åcopyå³è¾¹ç
help[i++] = arr[p1] < arr[p2] ? arr[p1++] : arr[p2++];
}
while (p1 <= m) {
help[i++] = arr[p1++];
}
while (p2 <= r) {
help[i++] = arr[p2++];
}
for (i = 0; i < help.length; i++) {
arr[L + i] = help[i];
}
return res;
}
// for test
public static int comparator(int[] arr) {
if (arr == null || arr.length < 2) {
return 0;
}
int res = 0;
for (int i = 1; i < arr.length; i++) {
for (int j = 0; j < i; j++) {
res += arr[j] < arr[i] ? arr[j] : 0;
}
}
return res;
}
// for test
public static int[] generateRandomArray(int maxSize, int maxValue) {
int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
for (int i = 0; i < arr.length; i++) {
arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
}
return arr;
}
// for test
public static int[] copyArray(int[] arr) {
if (arr == null) {
return null;
}
int[] res = new int[arr.length];
for (int i = 0; i < arr.length; i++) {
res[i] = arr[i];
}
return res;
}
// for test
public static boolean isEqual(int[] arr1, int[] arr2) {
if ((arr1 == null && arr2 != null) || (arr1 != null && arr2 == null)) {
return false;
}
if (arr1 == null && arr2 == null) {
return true;
}
if (arr1.length != arr2.length) {
return false;
}
for (int i = 0; i < arr1.length; i++) {
if (arr1[i] != arr2[i]) {
return false;
}
}
return true;
}
// for test
public static void printArray(int[] arr) {
if (arr == null) {
return;
}
for (int i = 0; i < arr.length; i++) {
System.out.print(arr[i] + " ");
}
System.out.println();
}
// for test
public static void main(String[] args) {
int testTime = 500000;
int maxSize = 100;
int maxValue = 100;
boolean succeed = true;
for (int i = 0; i < testTime; i++) {
int[] arr1 = generateRandomArray(maxSize, maxValue);
int[] arr2 = copyArray(arr1);
if (smallSum(arr1) != comparator(arr2)) {
succeed = false;
printArray(arr1);
printArray(arr2);
break;
}
}
System.out.println(succeed ? "Nice!" : "Fucking fucked!");
}
}
```
类似é¢ç®ï¼æ±ä¸ä¸ªæ°ç»ä¸çææéåºå¯¹ï¼ä¾å¦[3,1,7,0,2]éåºå¯¹ä¸ºï¼(3,1), (3,0), (3,2), (1,0), (70), (7,2)ãä¹å¯ä»¥åå©å½å¹¶æåºæ¥è§£å³ãå®è´¨å°±æ¯è¦æ±ä¸ä¸ªæ°å³è¾¹æå¤å°ä¸ªæ°æ¯èªèº«å°
> ä»ä¹æ ·çé¢ç®ä»¥åå¯ä»¥åå©å½å¹¶æåºï¼çº ç»æ¯ä¸ªæ°å³è¾¹(左边)æå¤å°ä¸ªæ°æ¯èªèº«å¤§ï¼æ¯èªèº«å°çãæ±è¿ç§æ°çæ°éçç
1.2 å¿«æ
>
1.2.1 Partionè¿ç¨
>
> ç»å®ä¸ä¸ªæ°ç»arrï¼åä¸ä¸ªæ´æ°numã请æå°äºçäºnumçæ°æ¾å¨æ°ç»ç左边ï¼å¤§äºnumçæ°æ¾å¨æ°ç»çå³è¾¹(ä¸è¦æ±æåº)ãè¦æ±é¢å¤ç©ºé´å¤æåº¦ä¸ºO(1)ï¼æ¶é´å¤æåº¦ä¸ºO(N)ãä¾å¦[5,3,7,2,3,4,1]ï¼num=3,æå°äºçäº3çæ¾å¨å·¦è¾¹ï¼å¤§äº3çæ¾å¨å³è¾¹
æè·¯ï¼è®¾è®¡ä¸ä¸ªå°äºçäºåºåï¼ä¸æ 为-1ã
1ã å¼å§éå该æ°ç»ï¼å¦æarr[i]<=num,å½åæ°ååºåä¸ä¸ä¸ªæ°äº¤æ¢ï¼åºåå峿©1ï¼i++
2ã arr[i] > num, ä¸åæä½ï¼i++
> ç»å®ä¸ä¸ªæ°ç»ï¼åä¸ä¸ªæ´æ°numã请æå°äºnumçæ°æ¾å¨æ°ç»ç左边ï¼çäºnumçæ¾ä¸é´ï¼å¤§äºnumçæ¾å³è¾¹ãè¦æ±é¢å¤ç©ºé´å¤æåº¦ä¸ºO(1)ï¼æ¶é´å¤æåº¦ä¸ºO(N)ã[3,5,4,0,4,6,7,2]ï¼num=4ãå®è´¨æ¯ç»å
¸è·å
°å½æé®é¢
æè·¯ï¼è®¾è®¡ä¸ä¸ªå°äºåºåï¼ä¸æ 为-1ã设计ä¸ä¸ªå¤§äºåºåï¼ä¸è¡¨ä¸ºarr.length,è¶çä½ç½®ã
1ã 妿arr[i]å½åä½ç½®çæ°==numï¼ i++ç´æ¥è·³ä¸ä¸ä¸ª
2ã 妿arr[i]å½åä½ç½®çæ°< numï¼å½åä½ç½®çæ°arr[i]åå°äºåºåçå³ä¸ä¸ªäº¤æ¢ï¼å°äºåºå峿©ä¸ä¸ªä½ç½®ï¼å½åä½ç½®i++
3ã 妿arr[i]å½åä½ç½®çæ°> numï¼å½åä½ç½®çæ°arr[i]ä¸å¤§äºåºåç左边ä¸ä¸ªäº¤æ¢ï¼å¤§äºåºå左移ä¸ä¸ªä½ç½®ï¼iåå¨åå°ä¸åå¤çï¼è¿éä¸åå¤çæ¯å 为å½åä½ç½®çæ°æ¯åä»å¤§äºåºå交æ¢è¿æ¥çæ°ï¼è¿æ²¡åæ¯è¾
4ãiå大äºåºåçè¾¹çç¸éï¼åæ¢æä½
1.2.2 å¿«æ1.0ï¼æ¯æ¬¡partionæå®ä¸ä¸ªä½ç½®
>
æè·¯ï¼å¨ç»å®æ°ç»ä¸åpartion,é宿°ç»æå³ä¾§çä½ç½®ä¸çæ°ä½ä¸ºnumï¼å°äºnumçæ¾å¨è¯¥æ°ç»ç左边ï¼å¤§äºnumçæ¾å¨è¯¥æ°ç»çå³è¾¹ã宿ä¹åï¼æè¯¥æ°ç»æå³ä¾§çæ°ç»numï¼äº¤æ¢å°å¤§äºnumåºåç第ä¸ä¸ªä½ç½®ï¼ç¡®ä¿äºäº¤æ¢åçnumæ¯å°äºçäºåºåçæåä¸ä¸ªæ°(该æ°ç´è³æåå¯ä»¥ä¿æå½åä½ç½®ä¸åï¼å±äºå·²ç»æå¥½åºçæ°)ï¼æè¯¥num左侧åå³ä¾§çæ°åå«è¿è¡åæ ·çpartionæä½(éå½)ãç¸å½äºæ¯æ¬¡partionæå®ä¸ä¸ªæ°çä½ç½®ï¼ä»£ç å®ç°quickSort1
1.2.3 å¿«æ2.0ï¼æ¯æ¬¡partionæå®ä¸æ¹ä½ç½®
>
æè·¯ï¼åå©è·å
°å½æé®é¢çæè·¯ï¼æarrè¿è¡partionï¼æå°äºnumçæ°æ¾å·¦è¾¹ï¼çäºæ¾ä¸é´ï¼å¤§äºæ¾å³è¾¹ãé彿¶æå°äºnumçåºåå大äºnumçåºååéå½ï¼çäºnumçåºåä¸åå¤çãç¸å½äºæ¯æ¬¡partionæå®ä¸æ¹æ°ï¼ä¸æ 记为ç¸ççæ°ã代ç å®ç°quickSort2
> 第ä¸çå第äºççå¿«ææ¶é´å¤æåº¦ç¸åO(N^2)ï¼ç¨æå·®æ
嵿¥è¯ä¼°ï¼æ¬èº«æåºï¼æ¯æ¬¡partionåªæå®äºä¸ä¸ªæ°æ¯èªèº«ï¼è¿è¡äºN次partion
1.2.4 å¿«æ3.0ï¼éæºä½ç½®ä½ä¸ºnumæ è®°ä½
>
==éæºéä¸ä¸ªä½ç½®iï¼è®©arr[i]åarr[R]交æ¢ï¼åç¨=arr[R]ä½ä¸ºæ è®°ä½ãå©ä¸çææè¿ç¨è·å¿«æ2.0䏿 ·ãå³ä¸ºæç»å
¸çå¿«æï¼æ¶é´å¤æåº¦ä¸ºO(NlogN)==
> 为ä»ä¹éæºéæ©æ 记为æ¶é´å¤æåº¦å°±ç±O(N^2)å为O(NlogN)?妿æä»¬éæºéæ©ä½ç½®é£ä¹å°±è¶åäºæ è®°ä½çå·¦å³ä¸¤ä¾§çéå½è§æ¨¡è¶åäºN/2ãé£ä¹æ ¹æ®masterå
¬å¼ï¼å¯ä»¥è®¡ç®åºç®æ³å¤æåº¦ä¸ºO(NlogN)ãå®è´¨ä¸ï¼å¨æä»¬éæ©éæºçnumæ¶ï¼æå·®æ
åµï¼æå¥½æ
åµï¼å
¶ä»åç§æ
åµçåºç°æ¦ç为1/Nã对äºè¿Nç§æ
åµï¼æ°å¦ä¸ç®åºçæ¶é´å¤æåº¦æç»æææ¯O(NlogN),è¿ä¸ªæ°å¦ä¸å¯ä»¥è¿è¡è¯æï¼æ¯è¾å¤æ
> ä¾å¦æä»¬çnuméæºå°æ°ç»å·¦ä¾§ä¸åä¹ä¸çä½ç½®ï¼é£ä¹masterå
¬å¼ä¸º
```math
T(N) = T((1/3)N) + T((2/3)N) + O(N)
```
> 对äºè¿ä¸ªéå½è¡¨è¾¾å¼ï¼masterå
¬å¼æ¯è§£ä¸äºçï¼masterå
¬å¼åªè½è§£å³åé®é¢è§æ¨¡ä¸æ ·çéå½ã对äºè¿ä¸ªéå½ï¼ç®æ³å¯¼è®ºä¸ç»åºäºè®¡ç®æ¹æ³ï¼å¤§è´æè·¯ä¸ºå设ä¸ä¸ªå¤æåº¦ï¼çè¿ä¸ªå
¬å¼æ¯å¦æ¶æäºè¿ä¸ªå¤æåº¦çæ¹å¼ï¼æ¯è¾éº»ç¦
1.2.5 å¿«æçæ¶é´å¤æåº¦ä¸ç©ºé´å¤æåº¦
>
> æ¶é´å¤æåº¦åè䏿æ¯ç§çå¤æåº¦
> 空é´å¤æåº¦ï¼O(logN)ã空é´å¤æåº¦äº§çäºæ¯æ¬¡éå½partionä¹åï¼æä»¬éè¦ç³è¯·é¢å¤ç空é´åéä¿åç¸çåºåçå·¦å³ä¸¤ä¾§çä½ç½®ãé£ä¹æ¯æ¬¡partionéè¦ç³è¯·ä¸¤ä¸ªåéï¼å¤å°æ¬¡partionï¼å®è´¨æ¯è¯¥é彿 被åäºå¤å°å±ï¼æ çé«åº¦ï¼æå¥½æåï¼æå¥½logN,æå·®Nãéæºéæ©numä¹åï¼ææä»ç¶æ¯æ¦çç´¯å ï¼æ¶æäºO(logN)ã
```Java
package class03;
public class Code03_PartitionAndQuickSort {
public static void swap(int[] arr, int i, int j) {
int tmp = arr[i];
arr[i] = arr[j];
arr[j] = tmp;
}
// partioné®é¢
public static int partition(int[] arr, int L, int R) {
if (L > R) {
return -1;
}
if (L == R) {
return L;
}
int lessEqual = L - 1;
int index = L;
while (index < R) {
if (arr[index] <= arr[R]) {
swap(arr, index, ++lessEqual);
}
index++;
}
swap(arr, ++lessEqual, R);
return lessEqual;
}
// arr[L...R] ç©è·å
°å½æé®é¢çååï¼ä»¥arr[R]åååå¼
// å°äºarr[R]æ¾å·¦ä¾§ çäºarr[R]æ¾ä¸é´ 大äºarr[R]æ¾å³è¾¹
// è¿åä¸é´åºåçå·¦å³è¾¹ç
public static int[] netherlandsFlag(int[] arr, int L, int R) {
// ä¸åå¨è·å
°å½æé®é¢
if (L > R) {
return new int[] { -1, -1 };
}
// å·²ç»é½æ¯çäºåºåï¼ç±äºç¨Råååè¿åRä½ç½®
if (L == R) {
return new int[] { L, R };
}
int less = L - 1; // < åº å³è¾¹ç
int more = R; // > åº å·¦è¾¹ç
int index = L;
while (index < more) {
// å½åå¼çäºå³è¾¹çï¼ä¸åå¤çï¼index++
if (arr[index] == arr[R]) {
index++;
// å°äºäº¤æ¢å½åå¼å左边ççå¼
} else if (arr[index] < arr[R]) {
swap(arr, index++, ++less);
// 大äºå³è¾¹ççå¼
} else {
swap(arr, index, --more);
}
}
// æ¯è¾å®ä¹åï¼æRä½ç½®çæ°ï¼è°æ´å°çäºåºåçå³è¾¹ï¼è³æ¤å¤§äºåºåææ¯çæ£æä¹ä¸ç大äºåºå
swap(arr, more, R);
return new int[] { less + 1, more };
}
public static void quickSort1(int[] arr) {
if (arr == null || arr.length < 2) {
return;
}
process1(arr, 0, arr.length - 1);
}
public static void process1(int[] arr, int L, int R) {
if (L >= R) {
return;
}
// L..Rä¸partition æ è®°ä½ä¸ºarr[R] æ°ç»è¢«åæ [ <=arr[R] arr[R] >arr[R] ]ï¼M为partionä¹åæ è®°ä½å¤å¨çä½ç½®
int M = partition(arr, L, R);
process1(arr, L, M - 1);
process1(arr, M + 1, R);
}
public static void quickSort2(int[] arr) {
if (arr == null || arr.length < 2) {
return;
}
process2(arr, 0, arr.length - 1);
}
public static void process2(int[] arr, int L, int R) {
if (L >= R) {
return;
}
// æ¯æ¬¡partionè¿åçäºåºåçèå´
int[] equalArea = netherlandsFlag(arr, L, R);
// 对çäºåºå左边çå°äºåºåéå½ï¼partion
process2(arr, L, equalArea[0] - 1);
// 对çäºåºåå³è¾¹ç大äºåºåéå½ï¼partion
process2(arr, equalArea[1] + 1, R);
}
public static void quickSort3(int[] arr) {
if (arr == null || arr.length < 2) {
return;
}
process3(arr, 0, arr.length - 1);
}
public static void process3(int[] arr, int L, int R) {
if (L >= R) {
return;
}
// éæºéæ©ä½ç½®ï¼ä¸arr[R]ä¸çæ°å交æ¢
swap(arr, L + (int) (Math.random() * (R - L + 1)), R);
int[] equalArea = netherlandsFlag(arr, L, R);
process3(arr, L, equalArea[0] - 1);
process3(arr, equalArea[1] + 1, R);
}
// for test
public static int[] generateRandomArray(int maxSize, int maxValue) {
int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
for (int i = 0; i < arr.length; i++) {
arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
}
return arr;
}
// for test
public static int[] copyArray(int[] arr) {
if (arr == null) {
return null;
}
int[] res = new int[arr.length];
for (int i = 0; i < arr.length; i++) {
res[i] = arr[i];
}
return res;
}
// for test
public static boolean isEqual(int[] arr1, int[] arr2) {
if ((arr1 == null && arr2 != null) || (arr1 != null && arr2 == null)) {
return false;
}
if (arr1 == null && arr2 == null) {
return true;
}
if (arr1.length != arr2.length) {
return false;
}
for (int i = 0; i < arr1.length; i++) {
if (arr1[i] != arr2[i]) {
return false;
}
}
return true;
}
// for test
public static void printArray(int[] arr) {
if (arr == null) {
return;
}
for (int i = 0; i < arr.length; i++) {
System.out.print(arr[i] + " ");
}
System.out.println();
}
// for test
public static void main(String[] args) {
int testTime = 500000;
int maxSize = 100;
int maxValue = 100;
boolean succeed = true;
for (int i = 0; i < testTime; i++) {
int[] arr1 = generateRandomArray(maxSize, maxValue);
int[] arr2 = copyArray(arr1);
int[] arr3 = copyArray(arr1);
quickSort1(arr1);
quickSort2(arr2);
quickSort3(arr3);
if (!isEqual(arr1, arr2) || !isEqual(arr2, arr3)) {
succeed = false;
break;
}
}
System.out.println(succeed ? "Nice!" : "Oops!");
}
}
```