' W3 N, B0 \2 ?8 r. K) ?( N0 }6 t d5 l9 r* ^! C4 h! y* f0 x2 n5 v( g+ W+ Y
下面以升序排列为例讲解过程:, u; a& i+ o, d, v$ V: _. w& D
5 d# m) Q+ H1 e- f7 A 原序列中可以分成两个序列,前面的是已排序的有序序列,用一个下标end标志该序列的尾,初始状态下end是0;后面的是还未排序的序列,用一个tmp变量暂存未排序序列首个元素,实际上是通过tmp = arr[end + 1] 实现的,也就是end指向的下一个元素。为什么要用变量暂存end的下一个元素呢?因为有可能涉及到元素后移,比如说,如果arr[end]要比tmp的值大,根据升序,应该把arr[end]的值后移对吧,那要是直接后移就会覆盖掉arr[end + 1],所以在移动前要先把元素暂存到tmp中。在每一次比较后end要递减一下,向前移动。要是tmp比arr[end]要大该怎么操作呢?这时候就说明找到要插入的位置了,把tmp的值插入到arr[end + 1]即可。 ' R P1 D7 ^5 b/ G- a8 c7 }8 m( W/ H4 w4 i. E
- R! G k e) h5 X. p, B5 m% p/ T0 k# X( I# T
可以看出,直接插入排序不可能只有一轮,像前面讲的过程就是一轮的过程,结果就是把未排序序列的首元素插入到了已排序序列中并保持原来的顺序,正如把刚摸到的一张牌按顺序插入到你的手牌中。实际上,一开始序列是全部未排序的,我们可以把第一个元素作为已排序元素,也就是说已排序序列一开始只有一个元素,end值就为0,后面的元素就全部都是未排序序列的了。 ; E B q0 o+ ^' m, K7 F" p7 M& p3 n" m3 b! I0 [
3 W* A9 M s' b% h5 P, o# o
. w9 ~/ z. V2 e* j6 i T* ] 也就是说要将整个原序列排好序的话要让已排序序列拓展至整个原序列,也就是end要不断增加到size - 1(size是数组元素个数),end值为size - 1时结束,所以还要外套一个循环。& m7 i# R8 {- [
0 U) y1 ], F5 i6 k4 H! avoid InsertSort(int* arr, int sz) 7 d3 u% b% v5 f$ `) c* W{ # |' X% I" P ]# ^3 z9 ?& `/ ~ assert(arr);( |# ]) t# r+ I, A* K& {
, o8 R( w8 M. X* k6 `* A
for (int i = 0; i < sz - 1; ++i)//i的取值就是end的取值,end最多取到sz-2算有效,当它取到sz-1时就该结束循环了. E- [; d1 [8 y/ I, m
{. C7 w ^ r/ K ^% L; J
//单轮排序 5 i; |: m' w" V% H; x int end = i;: L# x/ S3 n! [) d. c
int tmp = arr[end + 1]; , _, S; ?2 u6 b6 Z G, \# k while (end >= 0)//为什么要>=0呢?可不可以>0? . V& g, F5 [% J {! [# i! @8 [/ V& f
//要排升序 1 W; }3 K, j+ M& x5 u% `; d
if (tmp < arr[end]) ! C4 `. ]& y' ]" | { & b! w$ e6 k4 c arr[end + 1] = arr[end]; 2 h4 v8 ]+ z8 c --end; 0 K; H% g! q1 T } 3 W( N2 T6 N4 E/ @5 I else ) T' _5 X/ Y& ?5 J3 @: q; E) ]: j7 R { & e: w! m' a' {/ {; g& {
break; * g6 v% t4 Y! |3 Y3 d: q } # s" c5 t' m# [8 B
} 1 `- u. [* j' c# D5 l* N arr[end + 1] = tmp;5 @% H' N5 S( O8 H6 A3 ?
}; K0 B+ b g1 d6 s8 q( u7 y2 n
} % ?9 {( P$ u6 g6 D; P! A& S0 @5 B& B7 H3 _8 U, @
17 ?$ A) Q, p& z) E
2 + e4 w. {6 Z5 {2 D+ P! X3 / x; }" _8 m1 p* a3 S4 # \) i* k! @, Q' t* y: m57 t9 [2 h' W: W* D% D7 e: o4 V
6: p, A' O, {5 G: H2 M
7 ) D: D9 ]3 i/ Y& E, q" ^/ M8) A c2 N( V- x9 `- Z
9% n( r$ _: b/ V4 X
10 $ F% ~% K$ ~6 F: N# X11 0 G+ _! O* a- p+ w M% a' q6 C12 ' t% ^+ x r0 n9 c& s13 ( e( n/ c7 j r3 ^14/ ?$ a: m: {2 R+ X X
15; a' v! b5 l' S Y9 M* p5 S
16 + w8 e$ x5 s& R, p0 k; f178 Q( o' ^! b. i' C
18 % z1 I5 T* D8 U5 B# D19# l# i, c0 K/ w* e
20 , e6 @1 Z& b8 W21+ A3 w/ Q* }7 |8 Q, M8 l
224 J& \$ i/ E8 S5 L m+ G! D
237 _- y& W9 e0 F$ ]2 f! h; N
24 : N) g m1 W y# n ?7 X1 @) z& k25. v* S% L' [% c: o# ?
end可以为0,因为有可能tmp的值比arr[0]还小,那就得让原来的arr[0]后移,把tmp放到arr[0]的位置。; a" S) k% `& [2 v( d( `4 \
' t! H1 N5 y( K/ c7 _" g) u7 W0 g1 b+ Y) O, w5 ] K% g
9 ^8 O( t/ t0 c
直接插入排序的特性总结: / P5 ?, w( C# V4 F: R - d, }( R0 |7 E% h5 p- o: ^元素集合越接近有序,直接插入排序算法的时间效率越高,最好情况下(有序)时间复杂度为O(n) % z- E7 k8 W! O- d( [7 [8 G" d时间复杂度:O(n2) , V+ f. Y9 \- Y5 {空间复杂度:O(1),它是一种稳定的排序算法2 A! G& T& a& ~
稳定性:稳定- b" m1 y3 P7 R N v& Y: N
感谢观看,你的支持就是对我最大的鼓励~ , m2 G+ x: v, V1 }/ u————————————————0 l0 \2 D$ L1 B+ J2 R$ g
版权声明:本文为CSDN博主「桦秋静」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 Q! R0 B9 G" b
原文链接:https://blog.csdn.net/weixin_61561736/article/details/126796593 9 ~/ M' V0 j0 P. C) A. e # Y7 p; I Q. \; m: x/ G; ?0 L