Linux系统编程-进程-2 fork、execlp、execl、wait、waitpid

fork函数

pid_t fork();  创建一个子进程   返回值:-1 失败 、 等于0表示子进程  、 大于0表示父进程的id

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

pid_t getpid();  获取当前进程的id

pid_t getppid();  获取当前进程的父进程id

uid_t getuid();  获取当前进程实际用户id

gid_t getgid();  获取当前进程使用用户组id

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

int main(void)
{
    pid_t pid;

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

        while (1) {
            printf("I'm parent pid = %d, parentID = %d\n", getpid(), getppid());
            sleep(1);
        }
    } else if (pid == 0) {
        while (1) {
            printf("child  pid = %d, parentID=%d\n", getpid(), getppid());
            sleep(1);
        }
    }

    return 0;
}

父子进程共享

读时共享、写时复制。----全局变量

【重点】:父子进程共享:
1. 文件描述符(打开文件的结构体) 2. mmap 建立的映射区(进程间通信详解)

fork之后父进程先执行还是子进程限制性是不确定的。取决于内核所使用的调度算法。

exec函数族

使进程执行某一程序。成功无返回值,失败返回 -1

int execlp(const char *file, const char *arg, ...);        借助 PATH 环境变量找寻待执行程序

        参1: 程序名

        参2: argv0

        参3: argv1

        ...: argvN

        哨兵:NULL

该函数主要用来调用系统程序。如ls,date,cp,cat等

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

int main(int argc, char *argv[])
{
    printf("========================\n");
    
    pid_t pid = fork();
    if (pid == 0) {
        execl("/bin/ls", "ls", "-l", "-F", "-a", NULL);
        perror("execlp");
        exit(1);

    } else if (pid > 0) {
        sleep(1);
        printf("parent\n");
    }


    return 0;
}

int execl(const char *path, const char *arg, ...);        自己指定待执行程序路径。

加载一个进程,通过 路径+程序名 来加载。

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

int main(void)
{
	execl("/bin", "./ls", "-l", "-a", NULL);

	return 0;
}

//对比execlp ,如加载 " 命令带有 l F 参数
execlp("ls", "ls", "-l", "-F", NULL); //使用程序名在 PATH 中搜索。
execl("/bin/ls", "ls", "-l", "-F", NULL); //使用参数 1 给出的绝对路径搜索。

孤儿进程

    父进程先于子进终止,子进程沦为“孤儿进程”,会被 init 进程领养。

僵尸进程

    子进程终止,父进程尚未对子进程进行回收,在此期间,子进程为“僵尸进程”。  kill 对其无效。

wait函数

回收子进程退出资源, 阻塞回收任意一个。

一个进程在终止时会关闭所有文件描述符,释放在用户空间分配的内存,但是它的pcb还保留着,内核在其中保留了一些信息。如果正常终止则保存着退出状态,如果异常终止则保存着导致该进程终止的信号是哪个。这个进程的父进程可以调用wait或waitpid获取这些信息,然后彻底清楚掉这个进程。

pid_t wait(int *status)

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

        参数:(传出) 回收进程的状态。

        返回值:  成功: 回收进程的pid

                        失败: -1, errno

        函数作用1:    阻塞等待子进程退出

        函数作用2:    清理子进程残留在内核的 pcb 资源

        函数作用3:    通过传出参数,得到子进程结束状态


获取子进程正常终止值:

        WIFEXITED(status) --》 为真 --》调用 WEXITSTATUS(status) --》 得到 子进程 退出值。

获取导致子进程异常终止信号:

        WIFSIGNALED(status) --》 为真 --》调用 WTERMSIG(status) --》 得到 导致子进程异常终止的信号编号。


waitpid函数:    指定某一个进程进行回收。可以设置非阻塞。 

waitpid(-1, &status, 0) == wait(&status);

pid_t waitpid(pid_t pid, int *status, int options)

        参数:
                pid:指定回收某一个子进程pid

                        > 0: 待回收的子进程pid

                        -1:任意子进程

                        0:同组的子进程。

                status:(传出) 回收进程的状态。

                options:WNOHANG 指定回收方式为,非阻塞。

        返回值:

                > 0 : 表成功回收的子进程 pid

                0 : 函数调用时, 参3 指定了WNOHANG, 并且,没有子进程结束。

                -1: 失败。errno


总结:

    wait、waitpid    一次调用,回收一个子进程。

            想回收多个用while

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
进程编程是一种应用在Unix/Linux系统中的编程方法,其中一个重要的概念就是fork(分叉)。 在Linux中,fork是一个系统调用,它创建一个新的进程。通过fork,父进程可以生成一个与自己完全相同的子进程。父进程和子进程之间的唯一区别在于它们的进程ID不同,其他方面都是一模一样的。 fork的作用是允许一个进程创建一个与自己共享资源的进程,但是拥有独立的执行空间。这样可以避免不同进程之间相互影响和干扰。 在使用fork的过程中,操作系统会将父进程的所有信息(包括代码、数据、打开的文件描述符等)拷贝一份给子进程。然后,父进程和子进程会从fork调用的返回值中得到不同的结果。在父进程中,返回的是子进程进程ID;在子进程中,返回的是0。这样,父进程就可以根据fork的返回值来判断自己是父进程还是子进程,从而进行不同的操作。 在实际的编程中,可以通过fork来实现各种不同的功能。比如,可以通过fork创建一个子进程来进行并发执行,提高程序的运行效率。另外,可以通过fork来创建一个子进程来执行一段代码,然后通过父进程来等待子进程的结束并处理子进程的结果。 总之,进程编程中的fork是一个非常重要且常用的概念,它提供了创建和管理进程的能力,为程序的并发执行和资源的管理提供了可能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值