【linux进程控制(一)】进程终止--如何干掉一个进程?

简介: 【linux进程控制(一)】进程终止--如何干掉一个进程?

1. 前言

从这篇文章开始,将学习有关进程

控制的内容,包括:进程终止,进程等待

和进程的程序替换,内容不多,难度中等

请同学们耐心学习!

比起直接在界面输入CTRL+C
来终止一个进程外,我们更喜欢使用
一些函数接口(系统接口)来控制进程退出
一个进程的退出分为正常退出和异常退出

本章重点:

本篇文章着重讲解进程退出时
的三种场景以及常见的退出方法,
并且对比C库函数exit和系统调用
函数_exit的区别与联系.最后我们将
使用信号来模拟一些非正常退出的情况


2. 文章整体结构脉络

我们先把整篇文章的结构梳理出来

首先进程有三个退出场景:

  • 代码执行完,结果正确
  • 代码执行完,结果不正确
  • 代码异常终止了

第一,二种退出场景都是
属于正常退出的范围,正常退出
又有以下常见的方法:

  • 从main函数返回
  • 调用exit终止进程
  • 调用_exit终止进程

第三种退出场景是异常退出
异常退出往往是信号导致的,
我们熟悉的一个信号有:

  • CTRL+c,信号终止进程

全片文章将围绕以上内容做讲解!


3. 从main函数return进程

我们之前写C/C++代码时总会在写了

int main后写return 0,但是程序只能

return 0吗?答案是肯定不是!

先给出两个结论:

  1. 结论一:
  • 非main函数执行到return语句时
    代表此函数执行完毕!

  • main函数执行到return语句时
    代表此进程执行完毕!

  1. 结论二:
  • 程序正常执行完毕并且结果
    正确时返回0

  • 程序正常执行完毕但结果不正确
    时返回非0

有一个问题浮现在我们眼前,既然
结果不正确的返回值是非0,但是
非0有很多值,1,2,3,4,5等等,它们
分别有什么含义呢?不懂就去验证!

在验证前首先要明白一个函数:

它可以将错误码转换为错误字符串

int main()    
{    
    int i=0;    
    for(i=0;i<200;i++)                                                                                                                                                  
    {    
        printf("[%d]: %s\n",i,strerror(i));    
    }    
    return 0;    
}

不出所料,0对应成功!,

并且在134号错误以后,就是 未知错误了

查看最近进程的退出码:

使用指令: echo $?

写个代码,直接return来测试一下:

int main()
{
  return 66;
}


4. 使用库函数exit终止进程

exit的参数即为错误码,和main函数

的return值是一个意思

exit函数和return的区别:

  • return只有在main中使用时才
    代表此进程退出

  • exit函数在程序任一地方使用都
    可以直接退出程序,并且返回错误码

写一段代码验证一下:

void test1()
{
  exit(10);
}
void test2()
{
  exit(20);
}
int main()
{
  test1();
  exit(50);
  test2();
  return 0;
}

这里使用echo $?明显打印

出来的退出码是10!


5. 系统调用_exit和exit的区别

我们查看man的二号手册

可以看见_exit系统调用:

它和exit一样都是终止进程

并且_exit的参数也代表错误码

那么它们两个有什么区别呢?

我使用下面两段代码来验证:
一段用exit一段用_exit:

代码一:
  printf("你可以看见我吗?");
  sleep(1); //睡眠一秒
  exit(10);
代码二:
  printf("你可以看见我吗?");
  sleep(1); //睡眠一秒
  _exit(10);

下面两个图片对应运行结果:

直接看图片效果不是很好,这里建议

同学们自己去打一下这段代码

现象: 第一个打印了文字,而第二个没有打印

我们知道,printf打印的数据如果不使用
\n换行的话,数据会被存储到缓冲区里,
暂时不会打印出来,然而使用exit函数
结束进程后,缓冲区的数据被打印出来了
所以可以得出结论:exit函数会帮助我们
刷新缓冲区的数据,然而_exit函数不会

小思考:

既然exit是C语言提供的库函数
而_exit是操作系统提供的系统调用
_exit无法刷新缓冲区是不是说明缓冲区
压根就不在操作系统内?也就是说缓冲区
不由操作系统来维护,而是由C标准维护?

答案是,正确的!


6. 进程异常终止的场景

当我们在命令行输入CTRL+C传递

信号杀掉程序时,这是异常终止.

当程序中出现使用野指针或数组越界写入

时,程序会崩溃,崩溃也是进程异常结束

比如:

情况一:
int* p = NULL;
*p = 20;
情况二:
int a[10]={0};
a[11]=10;

此时运行程序后,程序会退出
这时候再去使用指令:echo $?
就没有意义了!

程序异常崩溃,退出码无意义
这是因为一般而言,退出码对应的
return语句还没执行到就已经崩溃了!


7. perror函数以及变量errno

errno是C语言中的一个全局变量

它里面存储的是最近的一个错误码
比如使用fopen函数打开文件时,
如果打开失败了,不仅仅文件指针fp
会被赋值为NULL,此时错误码errno
也会被系统自动赋值!

perror是C语言中常用的关于错误的函数

perror函数总是和errno搭配在一起
此函数会输出errno错误码对应的错误
信息,并且,perror的参数代表是哪个地方
有问题,是用户自己决定的!比如我们通常
这样来写一段代码:

FILE* fp = fopen("csdn.txt","r");
if(fp==NULL)
{
  perror("fopen");
  exit(1);
}

注意:打印出来的信息中,前面的fopen:
是用户输入的信息,后面的语句是errno
错误码对应的错误信息


8. 总结

进程终止话题是进程控制中最简单
的话题,学好进程终止可以提高我们
代码的可阅读性,可以把错误信息
展现的更加明了!

在本篇文章中出现的缓冲区概念只用于
区别exit和_exit的作用,更多关于缓冲区的
内容,我们将在Linux的基础IO中讲解!


🔎 下期预告:Linux进程等待 🔍


相关文章
|
6月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
250 67
|
5月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
138 16
|
5月前
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
108 20
|
4月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
86 0
|
4月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
118 0
|
4月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
84 0
|
4月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
82 0
|
7月前
|
Linux 数据库 Perl
【YashanDB 知识库】如何避免 yasdb 进程被 Linux OOM Killer 杀掉
本文来自YashanDB官网,探讨Linux系统中OOM Killer对数据库服务器的影响及解决方法。当内存接近耗尽时,OOM Killer会杀死占用最多内存的进程,这可能导致数据库主进程被误杀。为避免此问题,可采取两种方法:一是在OS层面关闭OOM Killer,通过修改`/etc/sysctl.conf`文件并重启生效;二是豁免数据库进程,由数据库实例用户借助`sudo`权限调整`oom_score_adj`值。这些措施有助于保护数据库进程免受系统内存管理机制的影响。
|
7月前
|
存储 Linux 调度
【Linux】进程概念和进程状态
本文详细介绍了Linux系统中进程的核心概念与管理机制。从进程的定义出发,阐述了其作为操作系统资源管理的基本单位的重要性,并深入解析了task_struct结构体的内容及其在进程管理中的作用。同时,文章讲解了进程的基本操作(如获取PID、查看进程信息等)、父进程与子进程的关系(重点分析fork函数)、以及进程的三种主要状态(运行、阻塞、挂起)。此外,还探讨了Linux特有的进程状态表示和孤儿进程的处理方式。通过学习这些内容,读者可以更好地理解Linux进程的运行原理并优化系统性能。
241 4
|
7月前
|
Linux Shell
Linux 进程前台后台切换与作业控制
进程前台/后台切换及作业控制简介: 在 Shell 中,启动的程序默认为前台进程,会占用终端直到执行完毕。例如,执行 `./shella.sh` 时,终端会被占用。为避免不便,可将命令放到后台运行,如 `./shella.sh &`,此时终端命令行立即返回,可继续输入其他命令。 常用作业控制命令: - `fg %1`:将后台作业切换到前台。 - `Ctrl + Z`:暂停前台作业并放到后台。 - `bg %1`:让暂停的后台作业继续执行。 - `kill %1`:终止后台作业。 优先级调整:
352 5