多进程:
-
什么是进程 1. 进程是程序在内存中的镜像 镜像:imag,把程序放在进程中就是进程 2. 进程是正在运行的程序 3. CPU 调度的最小单位。 进程是程序跑起来,调用在内存中,他在进程中构成的对象实例化,叫做进程 什么是程序 (应用):程序的集合,编译好的,可执行的二进制文件(Linux中为普通文件,不过有x权限:执行权限) 程序跑起来,调入到内存中,在内存中构建了一个对象或者是一个实例,这个实例叫做进程。
程序变成进程增加了很多概念:代码,内存空间,用户,组概念,打开的文件,占用的资源,合起来叫做一个进程。
4. 什么是线程
-
进程接口有哪些 1. fork():创建一个子进程 pid_t fork(void); //pid :process id,返回子进程的id 调用之后,父进程会拷贝自己一份,然后生成一份孩子进程 父亲和孩子运行在完全隔离/ 独立的内存空间 fork的时候内存会有一样的内容,但是内存写,和文件映射,不会产生影响 但是在fork的时后,有共同的内存空间,这一份数据是唯一的,当内存发生变化(写),拷贝才会真正发生,这个过程叫做写拷贝 孩子有自己的唯一的process ID 孩子不会继承父亲的内存锁,因为是两个独立的内存空间 进程的资源使用量和CPU的使用时间都会重置为0 孩子不会继承父亲未被执行的信号 孩子不会继承父亲相关联的记录锁,不会记录父亲的计时器 不会继承父亲的I/O操作 总之,子进程不会继承父进程的所有东西,除了父进程在调用子进程时,已经创建的空间。 返回值: 父进程返回的是孩子的pid, 孩子返回的是0,孩子可以调用getppid 获得父亲的pid, 也可以通过调用getpid获得自己的pid, 父进程没有办法获得自己生成进程的pid,除了第一次调用fork 时候的返回值。
#include "head.h"
int main() {
pid_t pid;
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
}
if (pid == 0) {
printf("Child Process!\n");
} else {
printf("Parent Process\n");
}
return 0;
}
输出结果为 但是一定先输出父进程么?或者说Parent进程一定先执行么? 未必,没有定论,系统会自动调动进程,不过极大概率都是父进程执行 同一时刻只有一个进程在CPU上跑,每个进程占用的时间是一定的,父进程生了孩子,还没用完,所以接着使用,子进程会等待父进程结束,所以大概率父进程先执行,如果父进程生了孩子,时间刚好用完,那么谁先执行就不一定了, 在pid = fork(),时候,生成孩子,孩子进程从该行代码之后执行
计算机中,父进程生孩子之后,让子进程执行任务,然后负责管理子进程
- wait() :
pid_t wait(int *wstatus); //等待状态发生变化,孩子发生状态改变时,会改变状态 wstatus 0,孩子正常死了,1,孩子出问题死了。。 wstatus 记录孩子的状态 释放孩子内存的不是父亲,而是父亲等待系统去释放,如果孩子死了,父进程不负责给孩子收尸,不等待,子进程变成僵尸进程,不占CPU,但是占系统资源,可以在top中看到 上述代码中的总父进程是zsh 如果孩子改变状态,调用立刻返回,否则,会阻碍程序的进行 创建僵尸进程
#include "head.h"
int main() {
pid_t pid;
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
}
int i = 0;
if (pid == 0) {
printf("Child Process! is : %d and Parent Process id is %d\n", getpid(), getppid());
} else {
printf("Parent Process id is %d\n", getpid());
while (1) sleep(100);
}
return 0;
}
ps -aux 查看 kill + 进程杀进程 杀死父进程,子进程也会直接被杀死。 单独kill 子进程无法杀死。
#include "head.h"
int main() {
pid_t pid;
int statu;
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
} else if (pid == 0) {
sleep(5);
printf("Hey I am the child\n");
return 0;
} else {
wait(&statu);
printf("The child is over and return %d\n", statu);
}
return 0;
}
生10个孩子,并排名
#include "head.h"
int main() {
pid_t pid;
pid_t child_pid[15] = { 0 };
pid_t father = getppid();
printf("father : %d\n", father);
int i;
for (i = 1; i <= 10; i++) {
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
}
if (pid == 0) {
printf("I am a child and rank the %2d and My pid is %d\n", i, getpid());
sleep(5);
printf("I am the %2d th child\n", i);
return 0;
}
}
sleep(10);
printf("Over\n");
return 0;
}
3. exec() :执行一个新的进程 新进程有不受父进程所管辖的全新进程 #include <unistd.h> extern char **environ; // 环境变量,一般写函数,都不会使用环境变量 int execl(const char *path, const char *arg, .../* (char *) NULL */); int execlp(const char *file, const char *arg ..., /* (char *) NULL */); int execle(const char *path, const char *arg, .../* (char *) NULL , char *envp[] */);
l : list, 有arg。。。。, 以一个空指针结束 p:从path路径找文件 v: 提供了一个数组,指向了一个以0结尾的字符串 e: 指定一个环境变量,通过envp 传入
不带p ,引入的必须是一个完整的路径,p模拟从shell 里面寻找
exec 家族函数,用一个新的进程映像,替换当前进程映像。 第一个参数,对于函数来说,是将要被执行文件的名字 *arg : 所有在execl(), execlp(), execle()函数中的arg 可以被认为arg0, arg1, arg2…argn
#include "head.h"
int main() {
pid_t pid;
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
}
if (pid == 0) {
execlp("vim", "vim", "4.exec.c", NULL);
printf("In End\n");
} else {
wait(NULL);
}
return 0;
}
execlp代码之后所有的在子进程中均不执行,因为子进程替换成vim 4.exec.c
##Problem
#include "head.h"
int main() {
pid_t pid;
if ((pid = fork()) < 0) {
perror("fork()");
exit(1);
}
int status;
if (pid == 0) {
printf("Child Process\n");
return 1;
} else {
wait(&status);
printf("Child is over, ret = %d\n", status);
}
return 0;
}
这段代码中,子进程返回值是1,但是在最后输出的时候,返回的值是256 为什么? 最低8位,在小端机中,二进制码为10000000 所以为256.
|