- 在线时间
- 1630 小时
- 最后登录
- 2024-1-29
- 注册时间
- 2017-5-16
- 听众数
- 82
- 收听数
- 1
- 能力
- 120 分
- 体力
- 563404 点
- 威望
- 12 点
- 阅读权限
- 255
- 积分
- 174244
- 相册
- 1
- 日志
- 0
- 记录
- 0
- 帖子
- 5313
- 主题
- 5273
- 精华
- 3
- 分享
- 0
- 好友
- 163
TA的每日心情 | 开心 2021-8-11 17:59 |
|---|
签到天数: 17 天 [LV.4]偶尔看看III 网络挑战赛参赛者 网络挑战赛参赛者 - 自我介绍
- 本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。
 群组: 2018美赛大象算法课程 群组: 2018美赛护航培训课程 群组: 2019年 数学中国站长建 群组: 2019年数据分析师课程 群组: 2018年大象老师国赛优 |
' W( g6 T% R8 D: E i
$ ]' G3 _2 J' F
神级基础排序——归并排序
! u, N6 Z: s0 G* `
8 K# H+ L* u b- X3 ]归并排序的介绍
6 w: _2 }: U/ {9 S5 {4 _
8 t" b4 y& B8 M. B; P# ?8 ?归并排序(英语:Merge sort,或Mergesort),是创建在归并操作上的一种有效的排序算法,其时间复杂度为O(N*logN)。1945年由约翰·冯·诺伊曼首次提出。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用,且各层分治递归可以同时进行。, F1 o1 ]5 F, _; [/ v
概念
* C( w7 G. ^; ?9 J3 }
" v: a6 j& p4 l- X4 i是利用递归与分治的技术将数据序列划分为越来越小的半子表,再对半子表排序,最后再用递归方法将排好序的半子表合并成越来越大的有序序列。
5 {1 T- ^% s4 I& f核心思想# `, b f& G4 n& E
3 I+ K. O$ W* ]3 E7 F! F
将两个有序的数列合并成一个大的有序的序列。通过递归,层层合并,即为归并。
9 I* ]1 y5 t' c3 R' a2 Y6 |) D% X' h# ?" u
4 }1 G- U& ^7 q% _9 h. v实现代码
; F1 w1 d; R% r, d) p: Uimport java.util.Arrays;$ ]; a2 x1 c. g" d% P7 Y# @* {
8 v1 _2 b1 T; z) b7 l, B/**! G0 F8 r# u, w. {4 _# S1 E
* @Author god-jiang/ e( I* A3 g; Z" q _
* @date 2020/1/13% c: }/ v1 ^2 i: W1 v& Z
*/0 ?6 m# s4 v2 f
//归并排序,时间复杂度为O(N*logN),空间复杂度为O(N)) s: |4 [7 }- | Y
public class MergeSort {
" d# O+ M* y4 N1 ?! c3 \- |' m/ `! I public static void MergeSort(int[] arr, int start, int end) {
2 j6 T% [5 A- O4 A //分治的结束条件1 {' Z1 d4 Z' D- S/ s. A, B. H# A8 q
if (start >= end) {
1 ~6 f3 O% q( [5 I7 p) U; F5 | return;
) N' i' P% Z( k, ~8 R }
3 g- S2 C8 _6 n3 m" s, [7 X //保证不溢出取start和end的中位数: @( z) b% C& F1 n8 j' ? s
int mid = start + ((end - start) >> 1);$ _. ]% W" q& [: H" r; d0 `0 [1 }8 B
//递归排序并且合并0 }: f# W5 b# P# K$ K( Y j
MergeSort(arr, start, mid);
) T# Q: Y; P$ P0 \7 @ MergeSort(arr, mid + 1, end);. w% K8 G: t7 N3 O5 I
Merge(arr, start, mid, end);' R k+ F7 s9 O# e& G
}
; g3 V& j8 Z- _( j6 P2 p, r) f; @1 q. x; @0 v
//合并
# L" y0 \) L+ O; e8 ?9 c3 f. f public static void Merge(int[] arr, int start, int mid, int end) {
7 P: a$ w6 g" N" u/ ~ int[] temp = new int[end - start + 1];
: j8 m2 X- l0 j2 E1 D1 x6 x: l/ } int p1 = start;5 o4 R+ j) a& v& o8 P& O# f8 n
int p2 = mid + 1;
. b$ U7 C8 _& r' L& l int p = 0;% T {& H" Q3 l8 k: @
while (p1 <= mid && p2 <= end) {
/ K: I" D7 ~# K8 Q% R% {/ a if (arr[p1] > arr[p2]) {
6 b4 e/ b+ r$ l" g6 U- J+ I temp[p++] = arr[p2++];; z: C. u8 _+ p; _
} else {
. S: U. k6 N& y. }& m9 `. |5 ] temp[p++] = arr[p1++];
$ d) @6 Q, i$ ? }3 R2 ^2 c7 \2 \
}
0 V; z; ^5 s8 f o# n while (p1 <= mid) {+ k( C- s0 H6 s6 K: `
temp[p++] = arr[p1++];6 H6 E) b) r2 f
}& s Y9 O& K. a% c) z) e8 G# G
while (p2 <= end) {+ k, p$ R( B* W+ C1 T" i
temp[p++] = arr[p2++];/ d6 O4 n) Y% R0 F& p& M& y/ ^2 b
}
! [) V, E; |, u3 q3 R6 C- Q* Y for (int i = 0; i < temp.length; i++) {% t1 {! J1 d& P+ ?1 m! p, x
arr[i + start] = temp;' Y. `" r8 M4 F, ]7 {! ~) x O6 D3 c. Y
}8 _* Q( }% q6 N( B' T+ B
}
; ]3 V9 x/ T3 e
+ M0 F, E/ i2 }1 c$ j1 w8 W public static void main(String[] args) {
' k3 S. _" c" w% L5 G" V9 K0 n/ M int[] a = {2, 4, 6, 1, 3, 7, 9, 8, 5};( S+ s: h& a+ m+ w- r% ~. ]# c; I
MergeSort(a, 0, a.length - 1);
% f9 X6 ?5 ?1 p& D' d- v System.out.println(Arrays.toString(a));3 W7 t1 O, c" X. [ E3 Q. v
}, c$ @. Y6 }" [+ k
}! W( v% U, _* P9 T6 P
$ M: M t! S9 {/ u8 D2 j, M5 B! m/ H7 i2 w1 P+ M. L
运行截图
* r5 `$ U3 k" b+ @' r+ P% P! G/ j) m! @6 V
+ V2 F" j5 u, y, P; L
/ t" Y W. |9 @) [5 c1、以上就是今天分享的归并排序,时间复杂度为O(N*logN),空间复杂度为O(N)
+ Y7 Z7 T' s0 A7 \" I1 q5 F$ O/ }2、归并排序的额外空间复杂度可以做到O(1),但是非常难,不需要掌握,有一篇论文”归并排序内部缓存法”可以做到* c, |+ r* ^/ l" r# D/ A
。! q: J( ]8 b8 U3 C
原文链接:https://blog.csdn.net/weixin_37686415/article/details/105180035
) ~4 Y0 g5 _3 C' j( h. s- e2 X: e* ?) {8 x$ S _
E2 S7 A9 ^* ?/ e2 f+ @& ?0 @
|
zan
|