二阶段-linux-进程-笔记

学习内容:

1.进程相关概念

2.创建进程函数fork的使用

3.vfork函数创建进程,与fork函数区别

4.进程退出(父进程等待子进程退出)

5.exec族函数

6.system函数

7.popen函数


一、

1.什么是程序,什么是进程,有什么区别?

①程序是静态概念,gcc xxx.c -o pro

磁盘中生成的pro文件,叫做程序

②进程是程序一次运行活动,通俗意思是程序跑起来了,系统就多了一个进程

2.如何查看系统中有哪些进程?

①使用ps指令查看

实际工作中,配合grep来查找程序中是否存在某一进程

ps -aux 查看所有进程

ps -aux|grep init  其中grep 是过滤  后面是仅查看init进程

②使用top指令查看,类似windows任务管理器

3.什么是进程标识符?

①每个进程都有一个非负整数表示的唯一ID,叫做pid,类似身份证

Pid = 0:称为交换进程(swapper) 

作用  ---  进程调度

Pid  = 1 :init 进程

作用 --- 系统初始化

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

//pid_t getpid(void);
//pid_t getppid(void);
int main(){
    pid_t pid;
    pid = getpid();
    printf("my pid is %d",pid);    
    return 0;
}

②编程调用getpid函数 获取自身的进程标识符,

getppid获取父进程的进程标识符

4.什么叫父进程,什么叫子进程

进程A创建了进程B

那么A叫做父进程,B叫做子进程,父子进程是相对的概念,理解为人类中的父子关系

5.C程序的存储空间是如何分配的?

二阶段-linux-进程-笔记

二阶段-linux-进程-笔记


二、创建进程函数fork的使用及应用场景

使用fork创建一个进程

pid_t fork(void);

fork函数调用成功,返回两次

返回值为0,  代表当前进程是子进程

返回值为非负数,  代表当前进程是父进程

调用失败, 返回-1

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

int main(){
    pid_t pid;
    pid = fork();
    
    if(pid>0){

        printf("this is father print,father pid = %d\n",getpid());
  } else if(pid == 0){
        pirntf("thid is child print,child pid = %d\n",get[id());
        
    }
        
    return 0;
}

fork()后代码段共享,父子进程均跑

#include <unistd.h>
#include<stdio.h>
int main(){
    pid_t pid;
    int date = 10;
    printf("father: id = %d\n",getpid());
    
    pid = fork();
    
    if(pid>0){
        printf("this is father pid :%d\n",getpid());

    }else if(pid == 0){
        printf("this is child pid :%d\n",getpid());
        date = date + 100;

    }
         printf("date = %d\n",date);   

        return 0;
}

输出结果:this is father pid = 6264

                  date = 10

                  this is child pid = 6265

                  date = 110 

全拷贝与写实拷贝

全拷贝,数据端 堆 栈 文件io流等

copy on write子进程对数据不改变,就共享变量,只有子进程改变数据,才对子进程的地址空间拷贝一份数据修改

fork应用场景

fork创建一个子进程的一般目的

(1)一个父进程希望复制自己,使父、子进程同时执行不同的代码段。这在网络服务进程中最常见的----父进程等待客户的服务请求。当这种情况到达时,父进程调用fork,使子进程处理此请求。父进程则继续等待下一个服务请求 的到达。

二阶段-linux-进程-笔记

(2)一个进程要执行一个不同的程序,这对shell是常见的情况。在这种情况下,子进程从fork返回后立即调用exec。

三、vfork函数也可以创建进程,与fork区别

关键区别一:

vfork直接使用父进程存储空间,不拷贝。

关键区别二:

vfork保证子进程先运行,当子进程调用exit退出后,父进程才执行。

四、进程退出

正常退出

1.Main函数调用return

2.进程调用exit(),标准c库

3.进程调用_exit()或者_Exit();属于系统调用

补充:

1.进程最后一个线程返回

2.最后一个县城调用pthread_exit

异常退出

1.调用abort

2.当进程收到某些信号时,如Ctrl+C

3.最后一个线程对取消(cancellation)请求做出响应

二阶段-linux-进程-笔记

②父进程等待子进程退出

(a.一个父进程希望复制自己,使父子进程同时执行不同的代码段。这在网络服务进程中是常见的——父进程等待客户端的服务请求。当这种请求到达时,父进程调用fork,使子进程处理此请求。父进程则继续等待下一个服务请求的到达。

    b.一个进程要执行一个不同的,这对shell是常见的情况。在这种情况下,子进程从fork返回后立刻调用exec)

父进程等待子进程退出并收集子进程的退出状态 wait(&status);父进程挂起状态等待子进程返回①wait();收集子进程状态,

waitpid(pid,&status,WNOHANG);子进程pid号,status(退出值),不挂起方式

父子进程一起走,不挂起,子进程结束状态被收集,变僵死进程

Waitpid

二阶段-linux-进程-笔记

--子进程退出状态不被收集,变成僵死进程(僵尸进程(Z+))

---孤儿进程:

     父进程如果不等待子进程退出,在子进程之前就结束了自己的”生命“,此时子进程叫做孤儿进程

      linux避免系统存在过多的孤儿进程,init进程收留孤儿进程,变成孤儿进程的父进程。getppid()

五、exec族函数

原文 https://blog.csdn.net/u014530704/article/details/73848573

exec族函数函数的作用:

我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不创建新进程,所以前后进程的ID并没有改变。

exec函数配合fork的使用:

实现功能,当父进程检测到输入为1的时候,建子进程把配置文件字段值修改掉。(简洁代码)

六、system函数

system相当于分装后的exec函数(简单粗暴,直接用)    sh -c ./***
相关函数
      fork,execve,waitpid,popen
表头文件
      #include<stdlib.h>
定义函数
      int system(const char * string);
函数说明
      system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。
返回值
      如果system()在调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果 system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。
附加说明
      在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。

七、popen函数

popen比system好处是  可以获取运行的结果

https://blog.csdn.net/libinbin_1014/article/details/51490568

学习产出:

提示:这里统计学习计划的总量
例如:
1、 技术笔记 2 遍
2、CSDN 技术博客 3 篇
3、 学习的 vlog 视频 1 个

上一篇:21_线程控制与通信


下一篇:MIT6.S081学习总结-lab6:Copy-On-Write Fork