Java數(shù)組排序原理

Arrays排序原理

Java Arrays排序原理

計數(shù)排序源碼

(short為例)

?
for (int i = left - 1; ++i <= right;count[a[i] - Short.MIN_VALUE]++);
?
for (int i = NUM_SHORT_VALUES, k = right + 1; k > left; ) {
 while (count[--i] == 0);
 short value = (short) (i + Short.MIN_VALUE);
 int s = count[i];
?
 do {
 a[--k] = value;
 } while (--s > 0);
}

插入排序源碼

(int為例)

 /*傳統(tǒng)插排(無哨兵Sentinel)
 * 遍歷
 *    循環(huán)向左比較(<左側(cè)元素——換位)-直到大于左側(cè)元素
 */
 for (int i = left, j = i; i < right; j = ++i) {
 int ai = a[i + 1];
 while (ai < a[j]) {
 a[j + 1] = a[j];
 if (j-- == left) {
 break;
 }
 }
 a[j + 1] = ai;
 }
} else {
 //如果一開始就是排好序的——直接返回
 do {
 if (left >= right) {
 return;
 }
 } while (a[++left] >= a[left - 1]);
?
 //優(yōu)化插排(以兩個為單位遍歷,大的元素充當哨兵引润,以減少小的元素循環(huán)向左比較的范圍)
 for (int k = left; ++left <= right; k = ++left) {
 int a1 = a[k], a2 = a[left];
?
 if (a1 < a2) {
 a2 = a1; a1 = a[left];
 }
 while (a1 < a[--k]) {
 a[k + 2] = a[k];
 }
 a[++k + 1] = a1;
?
 while (a2 < a[--k]) {
 a[k + 1] = a[k];
 }
 a[k + 1] = a2;
 }
 //確保最后一個元素被排序
 int last = a[right];
?
 while (last < a[--right]) {
 a[right + 1] = a[right];
 }
 a[right + 1] = last;
}
return;

快排源碼

(int為例)

// 快排閾值是286 其7分之一小于等于1/8+1/64+1
int seventh = (length >> 3) + (length >> 6) + 1;
?
// 獲取分成7份的五個中間點
int e3 = (left + right) >>> 1; // The midpoint
int e2 = e3 - seventh;
int e1 = e2 - seventh;
int e4 = e3 + seventh;
int e5 = e4 + seventh;
?
// 保證中間點的元素從小到大排序
if (a[e2] < a[e1]) { 
 int t = a[e2]; a[e2] = a[e1]; a[e1] = t; }
?
if (a[e3] < a[e2]) { 
 int t = a[e3]; a[e3] = a[e2]; a[e2] = t;
 if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }
}
if (a[e4] < a[e3]) { 
 int t = a[e4]; a[e4] = a[e3]; a[e3] = t;
 if (t < a[e2]) { a[e3] = a[e2]; a[e2] = t;
 if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }
 }
}
if (a[e5] < a[e4]) { 
 int t = a[e5]; a[e5] = a[e4]; a[e4] = t;
 if (t < a[e3]) { a[e4] = a[e3]; a[e3] = t;
 if (t < a[e2]) { a[e3] = a[e2]; a[e2] = t;
 if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }
 }
 }
}
?
// Pointers
int less  = left;  // The index of the first element of center part
int great = right; // The index before the first element of right part
?
//點彼此不相等——分三段快排期虾,否則分兩段
if (a[e1] != a[e2] && a[e2] != a[e3] && a[e3] != a[e4] && a[e4] != a[e5]) {
 /*
 * Use the second and fourth of the five sorted elements as pivots.
 * These values are inexpensive approximations of the first and
 * second terciles of the array. Note that pivot1 <= pivot2.
 */
 int pivot1 = a[e2];
 int pivot2 = a[e4];
?
 /*
 * The first and the last elements to be sorted are moved to the
 * locations formerly occupied by the pivots. When partitioning
 * is complete, the pivots are swapped back into their final
 * positions, and excluded from subsequent sorting.
 */
 a[e2] = a[left];
 a[e4] = a[right];
?
 while (a[++less] < pivot1);
 while (a[--great] > pivot2);
?
 /*
 * Partitioning:
 *
 *   left part           center part                   right part
 * +--------------------------------------------------------------+
 * |  < pivot1  |  pivot1 <= && <= pivot2  |    ?    |  > pivot2  |
 * +--------------------------------------------------------------+
 *               ^                          ^       ^
 *               |                          |       |
 *              less                        k     great
 */
 outer:
 for (int k = less - 1; ++k <= great; ) {
 int ak = a[k];
 if (ak < pivot1) { // Move a[k] to left part
 a[k] = a[less];
 /*
 * Here and below we use "a[i] = b; i++;" instead
 * of "a[i++] = b;" due to performance issue.
 */
 a[less] = ak;
 ++less;
 } else if (ak > pivot2) { // Move a[k] to right part
 while (a[great] > pivot2) {
 if (great-- == k) {
 break outer;
 }
 }
 if (a[great] < pivot1) { // a[great] <= pivot2
 a[k] = a[less];
 a[less] = a[great];
 ++less;
 } else { // pivot1 <= a[great] <= pivot2
 a[k] = a[great];
 }
 /*
 * Here and below we use "a[i] = b; i--;" instead
 * of "a[i--] = b;" due to performance issue.
 */
 a[great] = ak;
 --great;
 }
 }
?
 // Swap pivots into their final positions
 a[left]  = a[less  - 1]; a[less  - 1] = pivot1;
 a[right] = a[great + 1]; a[great + 1] = pivot2;
?
 // Sort left and right parts recursively, excluding known pivots
 sort(a, left, less - 2, leftmost);
 sort(a, great + 2, right, false);
?
 /*
 * If center part is too large (comprises > 4/7 of the array),
 * swap internal pivot values to ends.
 */
 if (less < e1 && e5 < great) {
 /*
 * Skip elements, which are equal to pivot values.
 */
 while (a[less] == pivot1) {
 ++less;
 }
?
 while (a[great] == pivot2) {
 --great;
 }
?
 /*
 * Partitioning:
 *
 *   left part         center part                  right part
 * +----------------------------------------------------------+
 * | == pivot1 |  pivot1 < && < pivot2  |    ?    | == pivot2 |
 * +----------------------------------------------------------+
 *              ^                        ^       ^
 *              |                        |       |
 *             less                      k     great
 *
 * Invariants:
 *
 *              all in (*,  less) == pivot1
 *     pivot1 < all in [less,  k)  < pivot2
 *              all in (great, *) == pivot2
 *
 * Pointer k is the first index of ?-part.
 */
 outer:
 for (int k = less - 1; ++k <= great; ) {
 int ak = a[k];
 if (ak == pivot1) { // Move a[k] to left part
 a[k] = a[less];
 a[less] = ak;
 ++less;
 } else if (ak == pivot2) { // Move a[k] to right part
 while (a[great] == pivot2) {
 if (great-- == k) {
 break outer;
 }
 }
 if (a[great] == pivot1) { // a[great] < pivot2
 a[k] = a[less];
 /*
 * Even though a[great] equals to pivot1, the
 * assignment a[less] = pivot1 may be incorrect,
 * if a[great] and pivot1 are floating-point zeros
 * of different signs. Therefore in float and
 * double sorting methods we have to use more
 * accurate assignment a[less] = a[great].
 */
 a[less] = pivot1;
 ++less;
 } else { // pivot1 < a[great] < pivot2
 a[k] = a[great];
 }
 a[great] = ak;
 --great;
 }
 }
 }
?
 // Sort center part recursively
 sort(a, less, great, false);
?
} else { // Partitioning with one pivot
 /*
 * Use the third of the five sorted elements as pivot.
 * This value is inexpensive approximation of the median.
 */
 int pivot = a[e3];
?
 /*
 * Partitioning degenerates to the traditional 3-way
 * (or "Dutch National Flag") schema:
 *
 *   left part    center part              right part
 * +-------------------------------------------------+
 * |  < pivot  |   == pivot   |     ?    |  > pivot  |
 * +-------------------------------------------------+
 *              ^              ^        ^
 *              |              |        |
 *             less            k      great
 *
 * Invariants:
 *
 *   all in (left, less)   < pivot
 *   all in [less, k)     == pivot
 *   all in (great, right) > pivot
 *
 * Pointer k is the first index of ?-part.
 */
 for (int k = less; k <= great; ++k) {
 if (a[k] == pivot) {
 continue;
 }
 int ak = a[k];
 if (ak < pivot) { // Move a[k] to left part
 a[k] = a[less];
 a[less] = ak;
 ++less;
 } else { // a[k] > pivot - Move a[k] to right part
 while (a[great] > pivot) {
 --great;
 }
 if (a[great] < pivot) { // a[great] <= pivot
 a[k] = a[less];
 a[less] = a[great];
 ++less;
 } else { // a[great] == pivot
 /*
 * Even though a[great] equals to pivot, the
 * assignment a[k] = pivot may be incorrect,
 * if a[great] and pivot are floating-point
 * zeros of different signs. Therefore in float
 * and double sorting methods we have to use
 * more accurate assignment a[k] = a[great].
 */
 a[k] = pivot;
 }
 a[great] = ak;
 --great;
 }
 }
?
 /*
 * Sort left and right parts recursively.
 * All elements from center part are equal
 * and, therefore, already sorted.
 */
 sort(a, left, less - 1, leftmost);
 sort(a, great + 1, right, false);
}

歸并排序前置處理

//判斷結(jié)構(gòu)是否適合歸并排序
int[] run = new int[MAX_RUN_COUNT + 1];
int count = 0; run[0] = left;
?
// Check if the array is nearly sorted
for (int k = left; k < right; run[count] = k) {
 if (a[k] < a[k + 1]) { // ascending
 while (++k <= right && a[k - 1] <= a[k]);
 } else if (a[k] > a[k + 1]) { // descending
 while (++k <= right && a[k - 1] >= a[k]);
 for (int lo = run[count] - 1, hi = k; ++lo < --hi; ) {
 int t = a[lo]; a[lo] = a[hi]; a[hi] = t;
 }
 } else { 
 //連續(xù)MAX_RUN_LENGTH(33)個相等元素,使用快排
 for (int m = MAX_RUN_LENGTH; ++k <= right && a[k - 1] == a[k]; ) {
 if (--m == 0) {
 sort(a, left, right, true);
 return;
 }
 }
 }
?
 //count達到MAX_RUN_LENGTH恳守,使用快排
 if (++count == MAX_RUN_COUNT) {
 sort(a, left, right, true);
 return;
 }
}
?
// Check special cases
// Implementation note: variable "right" is increased by 1.
if (run[count] == right++) { // The last run contains one element
 run[++count] = right;
} else if (count == 1) { // The array is already sorted
 return;
}

歸并排序源碼

(int為例)

byte odd = 0;
for (int n = 1; (n <<= 1) < count; odd ^= 1);
?
// Use or create temporary array b for merging
int[] b;                 // temp array; alternates with a
int ao, bo;              // array offsets from 'left'
int blen = right - left; // space needed for b
if (work == null || workLen < blen || workBase + blen > work.length) {
 work = new int[blen];
 workBase = 0;
}
if (odd == 0) {
 System.arraycopy(a, left, work, workBase, blen);
 b = a;
 bo = 0;
 a = work;
 ao = workBase - left;
} else {
 b = work;
 ao = 0;
 bo = workBase - left;
}
?
// Merging
for (int last; count > 1; count = last) {
 for (int k = (last = 0) + 2; k <= count; k += 2) {
 int hi = run[k], mi = run[k - 1];
 for (int i = run[k - 2], p = i, q = mi; i < hi; ++i) {
 if (q >= hi || p < mi && a[p + ao] <= a[q + ao]) {
 b[i + bo] = a[p++ + ao];
 } else {
 b[i + bo] = a[q++ + ao];
 }
 }
 run[++last] = hi;
 }
 if ((count & 1) != 0) {
 for (int i = right, lo = run[count - 1]; --i >= lo;
 b[i + bo] = a[i + ao]
 );
 run[++last] = right;
 }
 int[] t = a; a = b; b = t;
 int o = ao; ao = bo; bo = o;
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末槐雾,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子肉康,更是在濱河造成了極大的恐慌闯估,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,270評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件吼和,死亡現(xiàn)場離奇詭異涨薪,居然都是意外死亡,警方通過查閱死者的電腦和手機纹安,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,489評論 3 395
  • 文/潘曉璐 我一進店門尤辱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人厢岂,你說我怎么就攤上這事光督。” “怎么了塔粒?”我有些...
    開封第一講書人閱讀 165,630評論 0 356
  • 文/不壞的土叔 我叫張陵结借,是天一觀的道長。 經(jīng)常有香客問我卒茬,道長船老,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,906評論 1 295
  • 正文 為了忘掉前任圃酵,我火速辦了婚禮柳畔,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘郭赐。我一直安慰自己薪韩,他們只是感情好,可當我...
    茶點故事閱讀 67,928評論 6 392
  • 文/花漫 我一把揭開白布捌锭。 她就那樣靜靜地躺著俘陷,像睡著了一般。 火紅的嫁衣襯著肌膚如雪观谦。 梳的紋絲不亂的頭發(fā)上拉盾,一...
    開封第一講書人閱讀 51,718評論 1 305
  • 那天,我揣著相機與錄音豁状,去河邊找鬼捉偏。 笑死,一個胖子當著我的面吹牛泻红,可吹牛的內(nèi)容都是我干的告私。 我是一名探鬼主播,決...
    沈念sama閱讀 40,442評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼承桥,長吁一口氣:“原來是場噩夢啊……” “哼驻粟!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,345評論 0 276
  • 序言:老撾萬榮一對情侶失蹤蜀撑,失蹤者是張志新(化名)和其女友劉穎挤巡,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體酷麦,經(jīng)...
    沈念sama閱讀 45,802評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡矿卑,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,984評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了沃饶。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片母廷。...
    茶點故事閱讀 40,117評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖糊肤,靈堂內(nèi)的尸體忽然破棺而出琴昆,到底是詐尸還是另有隱情,我是刑警寧澤馆揉,帶...
    沈念sama閱讀 35,810評論 5 346
  • 正文 年R本政府宣布业舍,位于F島的核電站,受9級特大地震影響升酣,放射性物質(zhì)發(fā)生泄漏舷暮。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,462評論 3 331
  • 文/蒙蒙 一噩茄、第九天 我趴在偏房一處隱蔽的房頂上張望下面。 院中可真熱鬧,春花似錦绩聘、人聲如沸沥割。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,011評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至芹彬,卻和暖如春蓄髓,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背舒帮。 一陣腳步聲響...
    開封第一講書人閱讀 33,139評論 1 272
  • 我被黑心中介騙來泰國打工会喝, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人玩郊。 一個月前我還...
    沈念sama閱讀 48,377評論 3 373
  • 正文 我出身青樓肢执,卻偏偏與公主長得像,于是被迫代替她去往敵國和親译红。 傳聞我的和親對象是個殘疾皇子预茄,可洞房花燭夜當晚...
    茶點故事閱讀 45,060評論 2 355

推薦閱讀更多精彩內(nèi)容