首页 > 系统相关 >【Linux探索学习】第十七弹——进程终止:深入解析操作系统中的进程终止机制

【Linux探索学习】第十七弹——进程终止:深入解析操作系统中的进程终止机制

时间:2024-11-29 21:33:45浏览次数:9  
标签:return Linux 进程 exit printf 终止 include

Linux学习笔记:

https://blog.csdn.net/2301_80220607/category_12805278.html?spm=1001.2014.3001.5482

前言:

在操作系统中,进程终止是一个至关重要的阶段,它标志着进程的生命周期结束。进程终止可能是因为任务完成,也可能是因为异常或外部干预。本文将详细讲解操作系统中的进程终止相关知识,包括终止的原因、类型、实现方式、Linux系统中的具体操作,以及其影响和管理策略,并配以表格和代码示例,帮助全面掌握这一主题。


目录

一、什么是进程终止?

二、进程终止的主要原因

三、进程终止的类型

四、Linux中的进程终止实现

4.1 运行完毕且正常终止

4.1.1 使用return终止进程

4.1.2 使用exit终止进程

4.2 errno常量和strerror函数

4.2.1 strerror函数

4.2.2 errno常量

4.3 异常终止:abort

4.4 强制终止:kill

4.4 子进程资源回收:wait 和 waitpid

五、进程终止的影响

5.1 资源释放

5.2 僵尸进程

如何避免僵尸进程?

六、信号与进程终止

常见信号与作用

示例代码:捕获SIGTERM信号

七、进程终止的常见问题与解决

7.1 僵尸进程问题

7.2 非预期终止

八、总结


一、什么是进程终止?

进程终止(Process Termination)是操作系统中进程生命周期的最后一个阶段,意味着操作系统回收该进程的所有资源,包括内存、文件描述符、CPU时间等,使这些资源可以被其他进程使用。


二、进程终止的主要原因

进程可能因多种原因终止:

终止原因描述
正常终止进程完成所有任务后自然结束,例如程序执行到return语句或调用exit函数。
异常终止由于未处理的错误或异常导致进程终止,例如除以零、非法访问内存等。
外部干预进程被操作系统或其他进程强制终止,例如接收到SIGKILL信号。
父进程终止当父进程终止且子进程未被接管时,子进程可能成为孤儿进程,由initsystemd进程接管。
资源耗尽进程因超出系统资源限制(如内存、文件句柄等)被操作系统强制终止。

三、进程终止的类型

进程终止根据触发方式可以分为以下几类:

类型触发方式常见场景
正常终止调用exit()、返回主函数程序完成任务后自然结束。
异常终止未处理的错误或调用abort()例如访问非法地址、未处理的信号等。
强制终止外部进程调用kill()、操作系统干预父进程发送SIGKILL信号或管理员手动终止进程。
核心转储终止错误导致生成核心转储文件例如段错误(SIGSEGV)导致的异常。

一般进程终止的场景包含一下三种:

1. 代码运行完毕,结果正常

2. 代码运行完毕,结果不正常

3. 代码异常终止

下面我们会对上面的内容做出讲解


四、Linux中的进程终止实现

在Linux操作系统中,进程终止主要通过以下系统调用和信号实现:

4.1 运行完毕且正常终止

4.1.1 使用return终止进程

我们平时用的最多的方式就是return,我们先来看下面一个简单的代码

  #include<stdio.h>                    
     
  int main()   
  {   
      printf("This is a test message");
      return 0;
  }        

我们平时所写的代码main函数中一般都有一个返回值,那么这个返回值是干什么的呢?

main函数返回值是返回给进程看的,本质表示:进程运行完成时是否是正确的结果,如果是一般返回0,如果不是返回其它数字代表不同的退出信息(退出码)

我们可以通过这个指令打印退出码:

echo $?

4.1.2 使用exit终止进程

exit系统调用用于正常终止进程,并返回一个状态码给操作系统或父进程。

我们使用exit一般是在进程正常终止但没有正常执行的场景,或者是在合适的地方进行截停的场景,我们来看下面一段代码:

#include<stdio.h> 
#include<stdlib.h>                                                                                                                                                                                           
void print()      
{                 
    printf("hello linux\n");
    printf("hello linux\n");
    printf("hello linux\n");
    printf("hello linux\n");
    exit(20);     
    printf("hello linux\n");
    printf("hello linux\n");
}                 
int main()        
{                 
    print();      
    return 10;    
}                 

我们来看一下上面内容的执行结果和返回值:

我们发现返回值是exit中的返回值,并不是return的返回值,而且打印也只执行了四行,所以我们可以知道带有exit的进程,在执行到它时会直接返回,并不会再继续执行后面的内容,返回值也返回exit的返回值,这一点与return有较大差别


4.2 errno常量和strerror函数

4.2.1 strerror函数

上面我们提到我们可以通过不同的退出码来代表不同的错误信息,那么不同的退出码究竟各自代表什么信息呢?我们可以通过strerror函数来查看

比如我们来看一下退出码0到10所代表的信息:

#include<stdio.h>
#include<string.h>
int main()
{
    for(int i=0;i<=10;i++)
    {
        printf("%d: %s\n",i,strerror(i));
    }
    return 0;
}                

运行结果:

4.2.2 errno常量

上面我们讲到进程在退出是会有退出码,我们可以通过echo来查看退出码,那我们如何获取呢?

C/C++中其实还定义了一个叫errno的常量来记录错误码

所以我们就可以将errno常量与strerror函数结合使用,用errno来记录进程的错误码,然后传给strerror函数得到错误信息,比如下面的例子:

#include<stdio.h> 
#include<unistd.h>                                                                                                                                                                                           
#include<string.h>
#include<stdlib.h>
#include<errno.h>     //注意要带好头文件
int main()        
{                 
    int ret=0;    
    char *p=(char*)malloc(1000*1000*1000*4);    //这个扩容肯定会出错的,因为扩容空间太大了
    if(p==NULL)   
    {             
        printf("mallo error, %d:%s\n",errno,strerror(errno));   //errno会记录错误码,将它传到strerror中就可以得到错误信息
        ret=errno;    //将错误码作为返回值返回,从而让父进程得到返回信息
    }             
    else          
    {             
        printf("malloc success\n");
    }             
    return ret;   
}                 

运行结果:


4.3 异常终止:abort

4.2和4.3都牵扯到了信号的内容,这里我们主要还是以了解为主,后期我们会详细讲解信号的知识

abort函数用于非正常终止进程,通常在遇到不可恢复的错误时调用。

示例代码:abort

#include <stdlib.h>
#include <stdio.h>

int main() {
    printf("遇到严重错误,程序终止!\n");
    abort();  // 异常终止
    return 0;  // 不会被执行
}

调用abort会产生一个信号(SIGABRT),通常会生成一个核心转储文件供调试使用。


4.4 强制终止:kill

kill系统调用或命令用于向目标进程发送信号,例如SIGKILL信号会立即强制终止目标进程。

示例代码:kill

#include <signal.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    pid_t pid = fork();

    if (pid == 0) {
        // 子进程
        while (1) {
            printf("子进程正在运行: PID = %d\n", getpid());
            sleep(1);
        }
    } else if (pid > 0) {
        // 父进程
        sleep(5);
        printf("终止子进程: PID = %d\n", pid);
        kill(pid, SIGKILL);  // 发送SIGKILL信号
    }

    return 0;
}

执行结果:

我们发现子进程在被执行了五次后被终止掉了,对应的就是代码中执行了五次后会执行kill指令杀死进程


4.4 子进程资源回收:waitwaitpid

进程等待与回收我们会在下一篇详细讲解

当子进程终止后,其状态会保留在系统中,直到父进程回收。这种未被回收的子进程称为僵尸进程

示例代码:回收子进程资源

#include <sys/types.h>
#include <sys/wait.h>
#include <stdio.h>
#include <unistd.h>

int main() {
    pid_t pid = fork();

    if (pid == 0) {
        // 子进程
        printf("子进程开始: PID = %d\n", getpid());
        sleep(2);
        printf("子进程结束\n");
    } else if (pid > 0) {
        // 父进程
        int status;
        wait(&status);  // 等待子进程终止
        if (WIFEXITED(status)) {
            printf("子进程正常终止,状态码: %d\n", WEXITSTATUS(status));
        }
    }

    return 0;
}

执行结果:


五、进程终止的影响

5.1 资源释放

进程终止后,操作系统会回收以下资源:

  • 内存:代码段、数据段、堆栈等。
  • 文件描述符:关闭该进程打开的所有文件。
  • CPU时间:释放进程的时间片。

5.2 僵尸进程

当子进程终止但父进程未调用waitwaitpid回收其状态时,子进程会变成僵尸进程

如何避免僵尸进程?
  1. 父进程调用waitwaitpid回收子进程。
  2. 使用信号处理机制,如捕获SIGCHLD信号。

六、信号与进程终止

常见信号与作用

信号描述默认行为
SIGKILL强制终止进程,无法捕获或忽略。终止
SIGTERM请求终止进程,进程可以选择捕获或忽略。终止
SIGABRT异常终止进程,通常由abort触发。终止并生成核心转储
SIGCHLD子进程终止或停止时通知父进程。忽略
示例代码:捕获SIGTERM信号
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>

void handle_sigterm(int sig) {
    printf("接收到SIGTERM信号,进程终止\n");
    exit(0);
}

int main() {
    signal(SIGTERM, handle_sigterm);  // 注册信号处理函数

    while (1) {
        printf("程序正在运行...\n");
        sleep(2);
    }

    return 0;
}

七、进程终止的常见问题与解决

7.1 僵尸进程问题

问题:父进程未回收子进程,导致系统中出现僵尸进程。

解决

  • 调用waitwaitpid
  • 使用SIGCHLD信号处理函数自动回收。

7.2 非预期终止

问题:进程意外终止导致数据未保存。

解决:通过信号处理函数捕获终止信号,并在终止前完成必要的清理工作。


八、总结

进程终止是操作系统中管理资源的重要环节。通过本文的讲解,我们了解了进程终止的主要原因、类型以及Linux中的具体实现方式。进程终止不仅影响单个进程的生命周期,还对系统资源的利用和稳定性产生重要影响。

通过合理地使用exitkillwait等系统调用,以及信号机制,可以高效管理进程终止,避免僵尸进程问题,提高系统性能和可靠性。希望通过本文的详细分析和代码示例,你能更加深入理解操作系统的这一重要机制!

本篇笔记:


感谢各位大佬观看,创作不易,还请各位大佬点赞支持!!!

标签:return,Linux,进程,exit,printf,终止,include
From: https://blog.csdn.net/2301_80220607/article/details/143815200

相关文章

  • linux修改cst时区
    在Linux系统中,可以通过以下步骤将时区修改为CST(中国标准时间,GMT+8或称Asia/Shanghai):方法1:通过timedatectl命令修改(适用于大多数现代发行版)查看当前时区:timedatectl输出类似如下:Localtime:Fri2024-11-2910:00:00CSTUniversaltime:Fri2024-11-2902:00:00......
  • 【linux学习指南】Linux进程信号产生(二)软件中断
    文章目录......
  • 【Linux服务器】内存问题排查
    概述项目制作过程中经常出现内存问题,在该处对排查思路进行汇总,也对常见问题进行总结,以期待下一次遇到相似问题时可以快速排查,然后解决问题排查流程总结首先检查内存的整体情况使用工具htop和seme快速得知系统内存使用的全貌,识别高内存占用进程然后分析内存使用情况检查......
  • 【Linux服务器】网络问题排查思路总结
    1.总结常见问题无法连接到网络(例如无法访问互联网或内网)网络速度异常(如延迟、丢包、低带宽等)特定服务或端口无法访问排查网络问题思路2.问题与排查思路2.1检查网络接口通过IPa命令判断Ipv4地址以及接口是否正常工作(UP标识是否存在)ipa命令检查网络接口状态......
  • 计算机操作系统进程的描述与控制选择题
    1、正在执行的进程由于其时间片用完而被暂停运行,此时该进程应从运行态变为()。A、就绪态B、运行态C、等待态D、终止态2、某单处理器系统中若同时存在5个进程,则处于等待状态的进程最多可有()个。A、1B、0C、5D、43、一个进程退出等待队列而进入就绪队列,是因为进程()。A、获得了......
  • linux查看so库函数命令
    linux查看so库函数命令在Linux中,可以使用以下命令来查看共享对象(so)库的函数:nm命令:通过使用nm命令,可以列出so库中定义的所有符号,包括函数、变量等。使用以下命令:“nm-D“例如:nm-Dlibtest.so该命令将列出libtest.so库中的所有符号,其中包括函数名称。nm[option(s)]......
  • Linux常用基础指令-tar指令
      tar命令用于归档、压缩和解压文件。支持将多个文件或目录打包成一个文件,可以结合各种压缩算法生成高效的压缩文件格式(如.tar.gz、.tar.bz2等)。tar命令tar(tapetest的缩写)最初是为磁带设备设计的归档工具。随着时间的推移,它广泛应用于文件归档与压缩任务中。tar......
  • Linux常用命令之setfacl命令详解
    setfacl命令详解setfacl命令是Linux操作系统中用于设置文件或目录的访问控制列表(AccessControlLists,简称ACL)的工具。ACL提供了比传统Unix权限更细粒度的访问控制,可以为特定用户或用户组设置不同的权限。下面是关于setfacl命令的详细说明,包括基本语法、常用选项......
  • Linux常用命令之pvs命令详解
    pvs命令详解pvs命令是LVM(LogicalVolumeManager,逻辑卷管理器)工具集中的一个命令行工具,用于显示系统中所有物理卷(PhysicalVolumes,PV)的信息。LVM是一种磁盘管理技术,它允许用户将多个硬盘或分区组合成一个大的存储池,称为卷组(VolumeGroup,VG),然后从这个卷组中......
  • C++_Linux二进制格式
    C++调试工具GDB适合用于调试代码逻辑错误和程序崩溃,二者结合使用可以大大提高错误排查效率Valgrind和Memcheck更适合用于检查程序的内存问题,如内存泄漏、非法内存访问等。GFlags是VS中自带的内存检查工具gprof是一个GNU项目中的性能分析工具,用于分析C和C++程序的......