数学建模社区-数学中国

标题: 【基于C的排序算法】插入排序之直接插入排序 [打印本页]

作者: 杨利霞    时间: 2022-9-14 16:31
标题: 【基于C的排序算法】插入排序之直接插入排序
【基于C的排序算法】插入排序之直接插入排序9 b, P5 V* N' y! I' L# ?& g

0 k% F6 S8 i) C前言
9 s  J. _; d$ c本文基于C语言来分享一波笔者对于排序算法的插入排序中的直接插入排序的学习心得与经验,由于水平有限,纰漏难免,欢迎指正交流。8 W7 V0 V* m& Z( p; \
# T  ?# {7 q" N$ ~9 v! a
直接插入排序
: t/ N0 k. E$ H% C& j​ 直接插入排序是一种简单的插入排序法,其基本思想是 :
6 o/ }& \  L$ B* u$ [" c; G9 U
( M% K, p- d& T, X. J5 O​ 把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为
. M% i3 _# y, m* y4 F1 p止,得到一个新的有序序列 。
2 R( ~, p/ Z) _6 k6 r5 K​ 实际中我们玩扑克牌时,就用了插入排序的思想
9 `7 m- R8 C) O( }) _4 W& B3 n3 o2 s
1 E, x+ ?/ j5 a1 `# }2 r) p
& ^- Y+ \' i" A) w* F
/ F9 D& [7 @: Z8 o* L1 V​ 当插入第i(i>=1)个元素时,前面的array[0],array[1],…,array[i-1]已经排好序,此时用array的排序码与7 W) A; m4 ]* l
array[i-1],array[i-2],…的排序码顺序进行比较,找到插入位置即将array插入,原来位置上的元素顺序后移。$ g, b2 S0 m* N& F; Z; F

/ l) O6 S1 \0 n6 r# J升序排列的示例:
% U3 c9 `" V0 S2 ~  j: W# M' U. z, d& N
5 ^1 m' y3 W* o6 Q8 \: \. ?# r  B

5 [9 g8 E  Q& p下面以升序排列为例讲解过程:; \  _: v$ Z- ^; ^, O' |& o

* N1 `- T$ U+ H( Y' ?( f: m​ 原序列中可以分成两个序列,前面的是已排序的有序序列,用一个下标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]即可。, D+ j+ O% J; N* A+ Z

+ s- R% F9 ^) T9 w: @  f& Q  ^( L2 z) h* ~6 G& Q7 v6 i/ r
; C& \# A: N) A) z" y# V
​ 可以看出,直接插入排序不可能只有一轮,像前面讲的过程就是一轮的过程,结果就是把未排序序列的首元素插入到了已排序序列中并保持原来的顺序,正如把刚摸到的一张牌按顺序插入到你的手牌中。实际上,一开始序列是全部未排序的,我们可以把第一个元素作为已排序元素,也就是说已排序序列一开始只有一个元素,end值就为0,后面的元素就全部都是未排序序列的了。
5 ~* }9 @: y$ [: j4 Q, p
! T) v( t6 x6 K$ `+ S& y6 O: e% I' q) O: L4 R
8 p& p* @$ v! t. d- V
​ 也就是说要将整个原序列排好序的话要让已排序序列拓展至整个原序列,也就是end要不断增加到size - 1(size是数组元素个数),end值为size - 1时结束,所以还要外套一个循环。- t1 O- z5 v8 |& [# A' n+ v; F1 ^

6 c* K# `( j: S% Z4 ?void InsertSort(int* arr, int sz)( \8 ~$ a( U% ]
{
! \! Y: u$ Z$ {% a  i        assert(arr);
" R7 h( n, f( Q: A* k3 r; M4 m9 L4 l3 H6 c& A# i
        for (int i = 0; i < sz - 1; ++i)//i的取值就是end的取值,end最多取到sz-2算有效,当它取到sz-1时就该结束循环了
& z9 M3 \, e2 D- \/ n# y        {
& c8 U4 x7 U  e5 h. I3 Y( x                //单轮排序5 [/ |9 Q3 R( d
                int end = i;
4 A- o$ w3 T+ w8 C4 E; q8 ^  U                int tmp = arr[end + 1];! T& _, U6 n9 n8 y
                while (end >= 0)//为什么要>=0呢?可不可以>0?2 ]2 a3 |6 l# d1 |
                {
9 n2 A. I$ m2 t  Z                        //要排升序               
2 y! ~2 a: R5 A% P6 j/ I1 N. i                        if (tmp < arr[end])
' E  h2 G' m1 R* M                        {" e' a- z7 P9 _1 z
                                arr[end + 1] = arr[end];
3 U/ X( B+ P5 y! _2 M, Q9 e1 g- d                                --end;# E% ^8 N/ J- {$ Y/ G4 X. n
                        }
% e7 V  [/ q4 Z. `, b                        else
5 z& j+ `8 a3 @% O# F8 D                        {                               
* Q7 J/ Q* A" _" n                                break;4 I; u0 s6 R# Q& E8 s5 e4 H. z
                        }                3 {  P' h: ?6 z
                }
* q4 H& j, }( q  v! P- A- ?                arr[end + 1] = tmp;* ^: ~" o3 Q. O+ X; a' ~- g6 n! e8 V5 f
        }
2 q+ p( s4 t! O}% F# A9 F" c8 p" L( Y" Y* d' n

2 j8 \7 {& h1 F; T1
1 T3 S$ ?; Y0 \; h: L3 Q/ d2" e8 y+ q6 s! H' w- `$ \% u& R
3/ M5 B' v9 V8 W* \. @
4
. Q$ ]3 p' z3 V5 \! L  v53 P2 ?1 p! R3 }& ~0 x
6
5 Y% G* c& R/ c( Y! Z  X7' Y& R% N7 X) F2 B
84 W* q. [6 ?1 d) `0 y
9
. F2 S, P4 c. L# e7 Q# A7 d10
+ v" n! Y, `/ r# k. x" u" d4 F6 \9 J11
  {! m0 {# h7 O4 e12! {3 J: ?7 `1 h! g  h
13
7 j: g. ^, ?$ |. q14- R, Z6 _! b& f1 o' q( Q
15
4 J% T+ _1 W6 O) P, J3 e2 J16( H; ], j, @8 F3 w# ^* @, B: t
17
' a+ J5 X6 b1 a  W18
1 y0 U9 I' @; W! @. _  Q' c198 q+ ~1 L9 m8 h3 k' D- r- J
20
6 |. l& x! q' Z$ c3 ~21) a9 C' s- x# u% A
221 K7 }- S" M, T1 }0 g+ K6 a
23
- F- n' j  h$ x  c24
6 |" ^1 K7 k' C( b9 L' N25
7 w% `( s$ {1 `​ end可以为0,因为有可能tmp的值比arr[0]还小,那就得让原来的arr[0]后移,把tmp放到arr[0]的位置。! S- C! M' h( N

5 v% T4 d; \2 K
: q8 h7 _4 a2 c  K: C5 l7 E) L; x; n* ~: `
直接插入排序的特性总结:
+ Z8 A; a! @; v" N& r3 I, T) L
0 O$ k0 I4 L- M7 @$ v4 X1 k% O元素集合越接近有序,直接插入排序算法的时间效率越高,最好情况下(有序)时间复杂度为O(n)2 g% Y; R) q5 J7 I
时间复杂度:O(n2)6 L0 L+ q7 W+ c! y: s
空间复杂度:O(1),它是一种稳定的排序算法8 A& [# k% s: ?+ j! x" w# O: v
稳定性:稳定
- Y- s/ C& @( E* z9 P$ r感谢观看,你的支持就是对我最大的鼓励~
% J; I& M  w2 Y2 A- ]/ B9 @+ q" u' ]————————————————7 x5 X; b6 x" R0 H
版权声明:本文为CSDN博主「桦秋静」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
. _7 e7 t  F+ a原文链接:https://blog.csdn.net/weixin_61561736/article/details/126796593: Q. X( y! y1 S7 H$ G& d, k
! D; e, R+ V- s" S! }4 `: ^9 s

' f8 {$ X4 \% g7 h! U




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5