进程的结束

回顾上节:

父子进程的关系

子进程是父进程的副本

子进程获得父进程的数据段,堆,栈,正文段共享

在fork之后,一般情况下那个会先运行是不确定的,取决于操作系统

练习:

用进程实现文件拷贝

代码如下

复制代码
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>

//./a.out src dest 
int main(int argc, const char *argv[])
{
	if (argc != 3)
	{
		printf("Usgae: %s <src> <dest>\n",argv[0]);
		return -1;
	}

	pid_t pid = fork();

	int fd_s = open(argv[1],O_RDONLY);
	if (fd_s < 0 )
	{
		perror("open fail");
		return -1;
	}
	int fd_d = open(argv[2],O_WRONLY|O_CREAT|O_EXCL|O_TRUNC,0666);
	if (fd_d < 0 )
	{
		if (errno == EEXIST)
		{
			//perror("1 open fail");
			fd_d = open(argv[2],O_WRONLY);
		}else
		{
			perror("2 open fail");
			return -1;
		}
	}

	if (pid == -1)
	{
		perror("fork fail");
		return -1;
	}

	char buf[1024] = {0};
	struct stat st;
	stat(argv[1],&st);
	int len = st.st_size;
	int cnt = 0;
	if (pid > 0)
	{
		while (cnt < len/2)
		{
			int ret = read(fd_s,buf,sizeof(buf));
			cnt += ret;
			write(fd_d,buf,ret);
		}

	}else if (pid == 0)
	{
		lseek(fd_s,len/2,SEEK_SET);
		lseek(fd_d,len/2,SEEK_SET);
		while (cnt < len-len/2)
		{
			int ret = read(fd_s,buf,sizeof(buf));
			cnt += ret;
			write(fd_d,buf,ret);
		}
	}

	close(fd_s);
	close(fd_d);
	
	return 0;
}

进程创建好之后

1.子进程与父进程做的事情差不多 //子承父业

多任务 : 淘宝服务器 程序 ------创建子程序

2.父进程创建出子进程之后,子进程做的事情与父进程完全不同 //自力更生

shell程序 --- bash

exec函数族

作用:

启动(运行)一个新的程序

用一个新的进程的镜像取代(各个段)当前进程的镜像(各个段),但不包括pcb

函数族:

1、int execl(const char *path,const char *arg,.....);

l:list

path:要执行的文件路径 eg:"bin/ls"

arg:要执行的文件的名字 eg:"ls"

....:可执行文件所需要的参数 eg:对于cp指令,就需要原文件和目标文件

2、int execv(const char *path,const char *arg,.....);

V --- vector(向量)

3、int execlp(const char *file,const char *arg,.....);

带p的命令,除了第一个参数与上述不一样之外,其他参数相同。此时不需要写入文件的路径,只需要填入要执行的可执行文件的文件名 //可执行文件必须在系统的/bin目录中

4、int execvp(const char *file, char *const argv[]);

5、int execvpe(const char *file, char *const argv[],char *const envp[]);

env(environment)环境变量

一个一个字符串,只不过这个字符串的前面是,后面是

char *const envp[]:传递你要打印的环境变量

进程的执行

结束方式:

1.正常结束

1)main 中 return

2)exit() //库函数

c库函数,会执行io库的清理工作,关闭所有 的流,以及所有打开的文件。

注册清理函数(atexit)。

3)_exit,_Exit 会关闭所有的已经打开的文件,不执行清理函数。 //系统调用

//4) 主线程退出

//5)主线程调用pthread_exit

2.异常结束

6)abort()

7)signal kill pid

//8) 最后一个线程被pthread_cancle

相关推荐
qianpeng8977 小时前
水声匹配场定位原理及实验
算法
Johny_Zhao18 小时前
OpenClaw安装部署教程
linux·人工智能·ai·云计算·系统运维·openclaw
董董灿是个攻城狮19 小时前
AI视觉连载8:传统 CV 之边缘检测
算法
AI软著研究员1 天前
程序员必看:软著不是“面子工程”,是代码的“法律保险”
算法
FunnySaltyFish1 天前
什么?Compose 把 GapBuffer 换成了 LinkBuffer?
算法·kotlin·android jetpack
颜酱1 天前
理解二叉树最近公共祖先(LCA):从基础到变种解析
javascript·后端·算法
地平线开发者2 天前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮2 天前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者2 天前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考2 天前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习