Linux基础教程:9、linux进程管理(2)

news2025/7/9 9:17:23

前面我们讲到fork创建子进程,那么这一期我们接着讲创建进程之后如何调试以及插入其他进程、特殊进程、和进程如何退出;

同样我们写了一个C语言程序,但是在这个程序中是有两个进程,我们调试的时候只会选择一个进程调试,所以我们就需要学习多进程的调试;

1、同样是使用gdb但是不一样的是下面的设置:

set detach-on-fork off/on #设置是否分离线程调试
set follow-fork-mode parent #设置父子进程调试
info inferiors #打印调试信息,当前调试的进程以及可调式进程
inferior id #切换要调试的进程
detach inferiors id #分离调试

比如我们写了一个有父子进程代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{

	pid_t pid;
	int i;
	int num=0;
	pid=fork();
	
	for(i=0;i < 2;i++)
	{	
		if(pid==-1)
		{
			perror("fork error");
			exit(1);
		}
		else if(pid>0)
		{
			num+=2004;
			printf("花有重开日,人无再少年,id为:%d\n",getpid());
			printf("增加后的num:%d\n",num);
		}
		else if(pid==0)
		{
			num+=2004;
                	printf("花有重开日,人无再少年,id为:%d\n",getpid());
			printf("增加后的num:%d\n",num);
			exit(0);
		}
	}
	return 0;
}

我们先使用带有调试信息的编译进行编译,然后进入调试,这个文件名叫pipe.c所以:

 

 可以看到编译成功之后我们使用了gdb命令然后启动了调试,这里需要根据上面的命令设置一下:

我设置分离线程调试,而且是调试父进程,然后需要在打印的时候就断点:

 ​​

 我这里是23行,我在这里打了一个断点,然后启动调试:

 可以看到程序停在了这里,如果有循环的话我们点击下一步还是会停在这里,但是我们如果在紫禁城中也打断点它会不会进入子进程调试呢?

 这里我的子进程断点在29,那么我们打上断点开始调试试试:

 可以看到已经第二遍了还是没有进入子进程打印29行的内容,所以我们可以知道现在的调试程序是在对父进程进行调试,如果我们加上这一行代码他就会转换到子进程调试:

很明显我们输入:

set follow-fork-mode parent

程序就切换了进程调试;

那么多进程调试我就讲到这里,其他的功能就留给你们自己探索;

2、exec函数族

说这个是函数族不是函数肯定是有原因的,因为exec一个代表了六个函数:

execl(文件路径名,新进程名,NULL), #在程序中插入一个新进程,这个进程会运行参数1 
execlp(文件名,新进程名,NULL),  #与execl的作用一致,但是文件路径名会自动查找,不需要我们自己写好路径
execle(), 
execv(), 
execvp(), 
execve()

这里我就只讲解前两个其他的其实最终都是调用的第二个函数,我先讲一下第一个和第二个的        区别,第一个不会自己补齐路径的,但是第二个会自己找命令的;

那么他们的功能是什么呢?就是插入一个进程:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{

	pid_t pid;
	int i;
	int num=0;
	pid=fork();
	
	for(i=0;i < 2;i++)
	{	
		if(pid==-1)
		{
			perror("fork error");
			exit(1);
		}
		else if(pid>0)
		{
			num+=10;
			printf("花有重开日,人无再少年,id为:%d\n",getpid());
			printf("增加后的父进程num:%d\n",num);
		}
		else if(pid==0)
		{
			num+=10;
                	printf("花有重开日,人无再少年,id为:%d\n",getpid());

			execl("sy","sy",NULL);
			printf("增加后的子进程num:%d\n",num);

		}
	}
	return 0;
}

这个程序相比上一个就是多了一行代码:

execl("sy","sy",NULL);

这里的sy其实是我写的一个可执行文件,是编译之后的,他的源代码是:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{

	printf("fighting ,物联网2003! \n");
}

然后给他跑一下:

 原本执行的子进程到这行代码之后就跳到sy文件里面执行了,而且子进程在这个函数之后的代码都不会再执行了;那这个函数也就到这,execlp也差不多用法;

3、进程的退出

常见的进程退出有两种,一种是exit(),另一种叫就是_exit(),那么这两种有什么区别呢?

前者就是不刷新缓存区的退出,后者是会刷新缓存区的退出:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{
​
        printf("hello world!\n");
        prinft("hi");
        exit();
        return 0;
}

可以看到使用exit的时候两句话都是会打印的,也就是都执行了;但是如果我们换成_exit()就不一样了:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{
​
        printf("hello world!\n");
        prinft("hi");
        _exit(0);
        return 0;
}

4、特殊进程

这里我们就讲解两个进程,一个是僵尸进程一个是孤儿进程,僵尸进程在服务器维护当中是最不应该出现的,因为会占用资源,如果线程多的话还会让电脑崩溃,所以我们就需要阻止它的产生,而孤儿进程就是没有父进程的子进程,因为父子进程是异步运行的原因,所以存在父进程提前结束的情况,而子进程就没有人关闭,于是就一直运行:

int main()
{
​
        pid_t pid;
        int i;
        int num=0;
        pid=fork();
​
        for(i=0;i < 2;i++)
        {
                if(pid==-1)
                {
                        perror("fork error");
                        exit(1);
                }
                else if(pid>0)
                {
                        printf("parent pid:%d\n",getpid());
                }
                else if(pid==0)
                {
                        sleep(2);
​
                        printf("child pid:%d\n",getpid());
                }
        }
        return 0;
}

这里我们使用了一个操作,就是在创建了子进程之后,我们让其休眠2秒,之后父进程就会退出,它退出之后子进程就成了一个孤儿进程,结果就是子进程一直在运行,没有回收,而我们可以从窗口中看到没有显示原来的那个操作界面;

僵尸进程就是子进程已经结束,但是父进程却一直在执行没有回收,我们可以用下面的代码看到:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main()
{

        pid_t pid;
        int i;
        int num=0;
        pid=fork();

                if(pid==-1)
                {
                        perror("fork error");
                        exit(1);
                }
                else if(pid>0)
                {
			while(1)
			{

				printf("parent pid:%d\n",getpid());
				sleep(3);
			}
                }
                else if(pid==0)
                {
			sleep(2);

			printf("child pid:%d\n",getpid());
                }
        return 0;
}

这里我们就让父线程还没有苏醒的时候让子进程结束,于是我们就可以看到图中出现了一个z+,意思就是说僵尸进程还在运行;这时候我们结束掉父进程就可以结束僵尸进程了;

那么我们要如何来避免僵尸进程的产生呢?linux给出了wait()函数:

pid_t wait(int *wstatus);
​
 pid_t waitpid(pid_t pid, int *wstatus, int options);
 
 #include<sys/wait.h>
​

wait这个函数在父进程运行的时候会进行阻塞,如果发现子进程出现了僵尸进程的话就是执行,然后回收掉他,返回这个进程的id,如果没有的话就会返回-1 ,到时候正常退出就可以了;这里的参数是用来保存进程退出是的状态信息,因为我们只是为了处理僵尸进程,所以我们一般将参数设置为null就可以了;wait的使用需要包含一个头文件sys/wait.h

  while(1)
            {
​
                printf("parent pid:%d\n",getpid());
                sleep(3);
                ret=wait(NULL);
                if(ret==-1)
                {
                    exit(0);
                }
                else if(ret>0)
                {
                    printf("处理了一个僵尸进程!%d\n",ret);
                }
            }
  
我们在父进程的while循环里面添加这样的代码,然后结果就会截然不同: 

wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw== 编辑

原本会一直执行下去的父进程随着僵尸进程的产生随之就是退出了,是因为wait函数的调用,在产生了僵尸进程时候进行了处理然后打印了2099的编号,然后在没有了僵尸进程时候就执行了exit(0);

waitpid这个函数可以说是wait的升级版,他有三个参数,可以设置参数pid为-1,那么他的作用就是和wait一样的,他的显著特点就是不会阻塞线程,也就是在父进程运行的时候处理的子进程;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/18239.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

了解ixgbe网卡驱动— 驱动注册(纯代码分享)

1 ixgbe 网卡注册驱动 和大部分设备驱动一样&#xff0c;网卡驱动是作为一个 module 注册到 kernel 的 通过 module_init() -> ixgbe_init_module() -> pci_register_driver() 注册 ixgbe_driver 通过 module_exit() -> ixgbe_exit_module() -> pci_unregister_dr…

【仿牛客网笔记】项目进阶,构建安全高效的企业服务——置顶、加精、删除

添加依赖&#xff0c;去掉版本 实现置顶、加精的修改&#xff0c;删除 首先开发数据访问层&#xff0c;因为是对帖子的操作所以无论是置顶、加精最终是要修改帖子&#xff0c;先打开DiscussPostMapper增加修改的操作&#xff0c;一个修改类型&#xff0c;一个修改状态。 打…

[iOS]MonkeyDev安装

MonkeyDev官方安装文档&#xff1a;https://github.com/AloneMonkey/MonkeyDev/wiki/%E5%AE%89%E8%A3%85 1.安装HomeBrew 终端输入指令 /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)" 选择源后安装 更多安装方式可以参…

HTML+CSS+JS大作业:网站设计——家具装修公司(12页 bootstrap, 响应式)

⛵ 源码获取 文末联系 ✈ Web前端开发技术 描述 网页设计题材&#xff0c;DIVCSS 布局制作,HTMLCSS网页设计期末课程大作业 | 公司官网网站 | 企业官网 | 酒店官网 | 等网站的设计与制 | HTML期末大学生网页设计作业&#xff0c;Web大学生网页 HTML&#xff1a;结构 CSS&#…

如何使用VMware12PRO安装Mac OS

VMware安装参考我的这篇文章http://t.csdn.cn/kZ3oh &#xff08;有VMware安装包&#xff09; 准备的工具 unlocker链接: https://pan.baidu.com/s/1a1akxL_-JjNu70HJqJORZw?pwd9sjk 提取码: 9sjk Mac OS10.11CDR镜像链接: https://pan.baidu.com/s/1LIz0qFIL0Jg2M83oGqiDW…

RepVGG:让VGG风格的ConvNet再次伟大起来

引言 经典的卷积神经网络&#xff08;ConvNet&#xff09;VGG [31]在图像识别中取得了巨大的成功&#xff0c;其简单的架构由conv、ReLU和池化的堆栈组成。随着Inception [33&#xff0c;34&#xff0c;32&#xff0c;19]、ResNet [12]和DenseNet [17]的出现&#xff0c;许多研…

【Struts2】一_idea快速搭建struts2框架

文章目录什么是SSH框架&#xff1f;Struts2框架1、struts2的环境搭建1.1 创建web项目&#xff08;maven&#xff09;&#xff0c;导入struts2核心jar包1.2 配置web.xml&#xff08;过滤器&#xff09;&#xff0c;是struts2的入口&#xff0c;先进入1.3 创建核心配置文件struts…

C语言日记 36 类的组合

书P137: 如果声明组合类的对象时没有指定对象的初始值&#xff0c;自动调用无形参的构造函数&#xff0c; 构造内嵌对象时也对应调用内嵌对象的无形参的构造函数。 Q1:这里&#xff0c;对于“构造内嵌对象时也对应调用内嵌对象的无形参的构造函数”&#xff1b;他指的是什么…

STM32F429基于TouchGFX进行简单控制LED和显示ADC值

所需软件&#xff1a; CubeMX KEIL MDK ARM TouchGFX 首先配置外部时钟 配置时钟树&#xff0c;设置180MHZ 使能GPIO口如下&#xff0c;其中PA0用于LED 配置ADC通道 定时器TIM8触发 配置FMC和SDRAM,参数固定 使能DMA2D&#xff0c;参数如下&#xff1a; 配置LTDC 屏幕分…

JAVA反射

今天我们来讲一讲什么是java的反射机制,我们要了解一个新事物之前,我们应该首先的了解它的基本概念,那什么是反射呢? java的反射概念:JAVA反射机制是在运行状态中&#xff0c;对于任意一个类&#xff0c;都能够知道这个类的所有属性和方法&#xff1b;对于任意一个对象&#…

centos7 安装hadoop

文章目录一、将hadoop的压缩文件传递到虚拟机里面二、解压缩三、配置环境变量一、将hadoop的压缩文件传递到虚拟机里面 路径随意&#xff0c;只要你能到时候能找到压缩文件就行。 二、解压缩 这里我给解压到opt/module目录里(没有可以自己创建,主要是为了方便管理) tar -zx…

Java#15(集合)

一.集合和数组的区别 1.从长度方面: 数组的长度是固定的,而集合的长度不是固定的 2.从存储类型方面: 数组可以存储基本数据类型也可以存储引用数据类型,而集合能存储引用数据类型,若是想要存储基本数据类型要将其变成对应的包装类 创建一个集合 二.ArrayList成员方法 1.boole…

【Spring5】使用JdbcTemplate操作mysql数据库

文章目录1 JdbcTemplate简介及操作准备2 添加操作3 修改与删除操作4 数据库查询操作4.1 返回一个值4.2 返回对象4.3 返回集合5 批量操作5.1 批量添加与修改5.2 批量修改与批量删除写在最后1 JdbcTemplate简介及操作准备 Spring框架对JDBC进行封装&#xff0c;使用JdbcTemplate…

java计算机毕业设计ssm教师贴心宝的设计与实现

项目介绍 随着互联网技术的发发展,计算机技术广泛应用在人们的生活中,逐渐成为日常工作、生活不可或缺的工具,高校各种管理系统层出不穷。高校作为学习知识和技术的高等学府,信息技术更加的成熟,为教师开发必要的系统,能够有效的提升管理效率。近年来,高校规模不断扩大,同时在…

[附源码]SSM计算机毕业设计中达小区物业管理系统JAVA

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

c#入参使用引用类型为啥要加ref?

目录 ref修饰入参的常用场景引用类型添加ref的作用是啥&#xff1f;总结那什么是值&#xff0c;什么是引用&#xff1f;大体可以理解为堆栈的区别&#xff0c;在.net中大多数实例存在于托管堆栈中。struct&#xff0c;int32&#xff0c;int64&#xff0c;double&#xff0c;en…

mathtype符号显示不全的问题

目录前言参考问题解决方法补充最后前言 最近在使用Mathtype往Word里插入数学符号时&#xff0c;发现有些符号的间距过大&#xff0c;整个排版不太好看&#xff0c;于是上网查了一下&#xff0c;将方法记录下来。 参考 Word插入数学符号&#xff0c;行间距变大怎么办&#xf…

【408数据结构与算法】—堆排序(二十一)

【408数据结构与算法】—堆排序&#xff08;二十一&#xff09; 一、堆的定义 从堆的定义可以看出&#xff0c;堆实质是满足如下性质的完全二叉树&#xff0c;二叉树中任一非叶子结点均小于&#xff08;大于&#xff09;它的孩子结点 C语言代码实现 #include <stdio.h>…

python零基础入门教程(非常详细),从零基础入门到精通,看完这一篇就够了

前言 本文罗列了了python零基础入门到精通的详细教程&#xff0c;内容均以知识目录的形式展开。 第一章&#xff1a;python基础之markdown Typora软件下载Typora基本使用Typora补充说明编程与编程语言计算机的本质计算机五大组成部分计算机三大核心硬件操作系统 第二章&…

java计算机毕业设计ssm健达企业项目管理系统

项目介绍 随着经济的发展和信息技术的普及,国内许多健达企业都面临了重大的挑战。健达企业的管理流程、战略规划如果不能进行调整,极有可能面临淘汰的风险。特别是企业项目的处理,面对大量的人员信息和业务信息,如果不使用信息系统进行有效的管理和利用,那就会阻碍健达企业的发…