找回密码
 注册
关于网站域名变更的通知
查看: 38|回复: 0
打印 上一主题 下一主题

Linux进程退出常见方法

[复制链接]
  • TA的每日心情

    2019-11-20 15:16
  • 签到天数: 1 天

    [LV.1]初来乍到

    跳转到指定楼层
    1#
     楼主| 发表于 2024-8-21 14:24 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式

    EDA365欢迎您登录!

    您需要 登录 才可以下载或查看,没有帐号?注册

    x
         当一个进程结束了运行或在半途中终止了运行,那么内核就需要释放该进程所占用的系统资源。这包括进程运行时打开的文件,申请的内存等。# u1 [% _" d3 _9 K8 l

    : N: J$ I" [0 ]6 a. M/ j 7 E) f) z6 ~) f
    2 U  o% T8 A' A3 U# S
    进程退出0 s' _9 `& P4 n8 W2 \, ~! D( W
    ! f; {, b3 \/ N) @' a2 c
    Linux 下进程的退出分为正常退出和异常退出两种:+ t8 {  E; _2 k2 v
    / O) R+ c7 ]2 m- d, P" @2 I
    1.正常退出1 {) L) x$ O9 D- f
      j0 p! ]3 v5 B9 x
    a. 在main()函数中执行return 。
    , E' W) D+ b" Q" B0 j; S. B4 L6 f7 ?, m2 D
    b.调用exit()函数( h$ \, b; @/ N! `4 I/ ^! R

    : r: |0 m8 l, w' W' Xc.调用_exit()函数
    " q. D3 S, L# C' Y3 K* \
    , Z+ v  a2 ?# L" Z9 s, w2.异常退出) c0 m+ Q# K( N

    ! K; M* n/ }$ v$ j* M2 }a.调用about函数
    ! d; A7 r% v8 D5 Z
    6 g% \$ x# P3 ?# p, G6 \. _b.进程收到某个信号,而该信号使程序终止。
    , D1 R; V$ `$ T, H! V7 f7 r, L6 [' M) x$ S+ k% l. a
    不管是哪种退出方式,系统最终都会执行内核中的同一代码。这段代码用来关闭进程所用已打开的文件描述符,释放它所占用的内存和其他资源。
    9 A4 B- r5 T  `; I" K9 j0 g4 N- z. r7 o8 c9 S! Y4 f+ B; d* R
    几种退出方式的比较4 i. @0 ]2 H& F% b- |6 t3 `
    ' u3 a8 [* Z: A8 M* j6 n8 z
    1.exit和return 的区别:; q8 l9 e7 H& V$ [4 @9 f- t

    / H2 c: |. Z, p4 D7 Iexit是一个函数,有参数。exit执行完后把控制权交给系统2 F: N* h2 h8 t# N8 P9 C
    % B  T7 G% F1 d: K! u! }5 E$ \' \
    return是函数执行完后的返回。renturn执行完后把控制权交给调用函数。
    6 e  [5 `4 p  q4 e0 ]4 o2 W! m- [+ c# p  C" R
    2.exit和abort的区别:" U/ Z5 |# B! Z, [  `

    4 b: h% |( q+ I7 |exit是正常终止进程' a0 X( X8 l! x2 I

    : _" a7 n$ h+ ]; B7 j  @3 I2 fabout是异常终止。0 T( P9 J: T) t7 P

    7 o# b. r3 s0 I2 hexit()和_exit()函数
    7 w4 v' J3 G; h' ?' q( L3 G& V2 g( E0 s( h4 r
    exit和_exit函数都是用来终止进程的。当程序执行到exit或_exit时,系统无条件的停止剩下所有操作,清除各种数据结构,并终止本进程的运行。
    " A: N# r. W6 ]. @7 |$ X* P6 `( H3 ~% G: S) ]2 J, m) k
    exit在头文件stdlib.h中声明,而_exit()声明在头文件unistd.h中声明。 exit中的参数exit_code为0代表进程正常终止,若为其他值表示程序执行过程中有错误发生。
    & j/ j# F) Q: v/ z
    % Y1 o9 @1 K& a) E& cexit()和_exit()的区别
    / [2 M* a; f8 v: b# x8 d5 v3 E  T+ U/ j- H% @6 D( I5 i
    _exit()执行后立即返回给内核,而exit()要先执行一些清除操作,然后将控制权交给内核。
    - R- P9 d/ X8 m" L5 y
    ( x: [6 j1 R7 G, j' W调用_exit函数时,其会关闭进程所有的文件描述符,清理内存以及其他一些内核清理函数,但不会刷新流(stdin, stdout, stderr ...). exit函数是在_exit函数之上的一个封装,其会调用_exit,并在调用之前先刷新流。2 b* C$ h4 ?- h0 d

    : M$ q$ |+ a7 zexit()函数与_exit()函数最大区别就在于exit()函数在调用exit系统之前要检查文件的打开情况,把文件缓冲区的内容写回文件。由于Linux的标准函数库中,有一种被称作“缓冲I/O”的操作,其特征就是对应每一个打开的文件,在内存中都有一片缓冲区。每次读文件时,会连续的读出若干条记录,这样在下次读文件时就可以直接从内存的缓冲区读取;同样,每次写文件的时候也仅仅是写入内存的缓冲区,等满足了一定的条件(如达到了一定数量或遇到特定字符等),再将缓冲区中的内容一次性写入文件。这种技术大大增加了文件读写的速度,但也给编程代来了一点儿麻烦。比如有一些数据,认为已经写入了文件,实际上因为没有满足特定的条件,它们还只是保存在缓冲区内,这时用_exit()函数直接将进程关闭,缓冲区的数据就会丢失。因此,要想保证数据的完整性,就一定要使用exit()函数。
    5 h0 Q( ?! V2 U$ q
    5 J/ X+ T/ }% C通过一个函数实例来看看它们之间的区别:5 K+ o& G5 \8 ^* ?% ^9 o
    3 e# i  s: A& Z
    函数实例1 : exit.c- `7 m3 h$ R+ ?0 A! |

    - f8 w7 Z+ b+ |- ]1 t$ u2 }! x#include#includeint main(){printf("using exit----\n");printf("This is the content in buffer\n");exit(0);}1 s" ~* X5 X, c. y2 Z

    8 s' d: @7 @# Y* x执行结果为:6 n, i. w! ]  V5 ]3 E

    8 t; Y) `- a" @3 B) o! c  Nusing exit----This is the content in buffer
    9 Z6 F- n6 H  m  F' Z6 }% p; _2 e/ S! K+ E+ Z* e! |4 l' a
    函数实例2:_exit.c4 L, ?4 \) y1 z- ^( F. S$ K# L
    2 ~! X2 A0 O3 L: r
    #include#includeint main(){printf("using _exit--\n");printf("This is the content in buffer");_exit(0);}
    6 r+ j* }0 |3 d$ \1 t
    ( z5 \0 U" j  @& R; a% z执行结果为 :
    9 ^6 T  @! A! j) J( X  d! z3 q: ?  B# j& B' F2 n
    using _exit--: F# Y% ^' E& Y

    # r+ l7 f8 [5 @" A! lprintf函数就是使用缓冲I/O的方式,该函数在遇到“\n”换行符时自动的从缓冲区中将记录读出。所以exit()将缓冲区的数据写完后才退出,而_exit()函数直接退出。
    / y0 T5 R. W$ K* w" b' ?. D7 M/ Z) A8 J+ H# W
    大家也可以把函数实例2中的printf("This is the content in buffer");改为printf("This is the content in buffer\n")(即在printf中最后加一个\n看运行结果是什么,为什么会产生这样的结果呢?)5 r6 Q- _; `+ x0 U7 W9 r3 X; X

    ( V, @7 }" r7 E" m) a! H父子进程终止的先后顺序不同会产生不同的结果
    * x7 b1 M; u8 A, k
    : R' e, x" |2 F2 }0 t5 q1.父进程先于子进程终止:+ m7 B$ f8 T$ d( c" Q

    8 [! z5 L' s) V2 K  @此种情况就是我们前面所用的孤儿进程。当父进程先退出时,系统会让init进程接管子进程 。
    7 X, p3 Y8 j' I; f- v, o4 Z: B( C* E* }8 X7 d; p
    2.子进程先于父进程终止,而父进程又没有调用wait函数1 S, x/ S6 S6 g, U/ I" s: ^0 l
    + `+ Q4 h' @4 H$ X3 A6 ]- n2 Z$ p$ p
    此种情况子进程进入僵死状态,并且会一直保持下去直到系统重启。子进程处于僵死状态时,内核只保存进程的一些必要信息以备父进程所需。此时子进程始终占有着资源,同时也减少了系统可以创建的最大进程数。8 I' N) B/ `/ s! z1 E7 u2 z
    , ^, W- B$ v5 d+ G( _
    什么是 僵死状态呢?/ t0 v( F$ O% g
    5 Z/ j4 r$ U& X8 C. O
    一个已经终止、但是其父进程尚未对其进行善后处理(获取终止子进程的有关信息,释放它仍占有的资源)的进程被称为僵死进程(zombie)。+ S$ q' _. N& M8 [, g, w& r: _8 @' v

    0 \1 v  }0 X" X9 p$ A, O0 b3.子进程先于父进程终止,而父进程调用了wait函数5 }; d7 I) R- C
    4 i- l; y/ C6 ?8 {" V/ o- s" M" g
    此时父进程会等待子进程结束。
    & ]* C* B$ [. l* X3 L4 ^8 c6 G
      ^$ I# Q; h! }# H0 g9 K. h! u7 P4 J% c. t

    , Q; V$ F$ c* f. G# f) n0 F7 F* x5 [1 s8 \
    3 t+ b3 k3 P8 g2 X  ?
    您需要登录后才可以回帖 登录 | 注册

    本版积分规则

    关闭

    推荐内容上一条 /1 下一条

    EDA365公众号

    关于我们|手机版|EDA365电子论坛网 ( 粤ICP备18020198号-1 )

    GMT+8, 2025-8-11 17:59 , Processed in 0.140625 second(s), 27 queries , Gzip On.

    深圳市墨知创新科技有限公司

    地址:深圳市南山区科技生态园2栋A座805 电话:19926409050

    快速回复 返回顶部 返回列表