创建子进程

fork

创建一个子进程。

  • SYNOPSIS
    #include <unistd.h>
    
    pid_t fork(void);
    
  • RETURN VALUE
    一旦调用成功父进程中的 fork 会返回子进程 PID,而子进程的 fork 会返回0;调用失败父进程中 fork 返回-1,没有子进程创建出来。【注意】:不是 fork 函数能返回两个值,而是 fork 后,fork 函数变为两个,父子需各自返回一个。

创建单个子进程模型

【注意】:父进程创建子进程后,两者竞争 cpu,如要保证子进程先于父进程完成可以在父进程处理分支中加入 sleep 语句,阻塞一段时间让子进程先处理完毕。

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

int main(int argc, char *argv[])
{
	pid_t pid = 0;
	
	pid = fork();
	if (pid == -1) {
		// 返回值错误判断
	} else if (pid == 0) {
		// 子进程处理
	} else {
		// 父进程处理
	}
	
	return 0;
}

循环创建N个子进程模型

【注意】:

  • 一次 fork 调用可以创建一个子进程,可以通过 for 循环进行创建指定数目的子进程;
  • 在 for 循环中需要加入对子进程的判断提前 break 结束 for 循环,否则会产生子进程继续创建子进程的情况,循环创建了 (2^N)-1 个子进程,而不是N个子进程;
  • 这些子进程几乎是同时创建出来的,与父进程共同竞争 cpu,若要保证顺序可以加入 sleep 操作阻塞一段时间;
  • 在 shell 中启动进程创建子进程并且没有阻塞情况发生时,shell 等启动的进程返回后也会存在同其它子进程竞争 cpu 的情况(shell 本身也是进程),所以在打印时可能会看到子进程还没结束运行提前出现了 shell 命令行,如果加入 sleep 操作可避免此现象;
  • 如果要跟踪第 i 个子进程进行某些特殊处理,可以通过循环因子进行判断;
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char *argv[])
{
	pid_t pid;
	int idx;

	for (idx=0; idx<N; idx++)
	{
		pid = fork();
		if (idx ==-1) {
			// 返回值错误判断
		} else if (pid == 0) {
			// 子进程处理
			break;
		} else {
			//父进程处理
		}
	}

	//sleep(idx);
	if (idx < 5) {
		// 子进程处理
	} else {
		// 父进程处理
	}

	return 0;
}

进程共享

【fork后差异】:

  • 相同:全局变量、.data、.text、栈、堆、环境变量、用户ID、宿主目录、进程工作目录、信号处理方式…
  • 不同:进程ID、fork返回值、.父进程ID、进程运行时间、.闹钟(定时器) 、未决信号集…

似乎,子进程复制了父进程0-3G用户空间内容,实际上并非是完全拷贝,父子进程间遵循【读时共享写时复制】的原则,这样做能够节省开销。
【父子进程共享】:

  1. 文件描述符(打开文件的结构体);
  2. mmap建立的映射区(可用于进程间通讯);

【注意】:

  • 普通全局变量不能用于进程间通信,fork 后子进程继承了父进程的变量,但是之后的读写都是独立的。

进程相关接口

getpid/getppid

获取当前进程ID/获取当前进程的父进程ID。

  • SYNOPSIS
    #include <sys/types.h>
    #include <unistd.h>
    
    pid_t getpid(void);
    pid_t getppid(void);
    
  • RETURN VALUE
    返回当前进程ID/返回当前进程的父进程ID。

getuid/geteuid

获取当前进程【实际用户ID】/获取当前进程【有效用户ID】。

  • SYNOPSIS
    #include <unistd.h>
    #include <sys/types.h>
    
    uid_t getuid(void);
    uid_t geteuid(void);
    
  • RETURN VALUE
    返回当前进程实际用户ID/返回当前进程有效用户ID。

getgid/getegid

获取当前进程【使用用户组ID】/获取当前进程【有效用户组ID】。

  • SYNOPSIS
    #include <unistd.h>
    #include <sys/types.h>
    
    gid_t getgid(void);
    gid_t getegid(void);
    
  • RETURN VALUE
    返回当前进程【使用用户组ID】/返回当前进程【有效用户组ID】。
  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值