OSLab多进程
日期:2019/3/23
内容:Linux下与多进程相关的函数。
进程基本知识
- 定义
应用程序关于某数据集合上的一次运行活动。
- 特点
·操作系统进行资源分配和调度的基本单位
·进程是程序的一次执行过程。进程是动态的,程序是静态的。
·同一程序同时运行于若干个数据集合上,该程序将对应与若干个不同的进程。
·每个进程拥有独立的地址空间。地址空间包括代码段、数据段和堆栈段。
·进程之间的地址空间是隔离的。一个进程崩溃不会影响另一个进程,一个进程崩溃不会影响到操作系统。(一个程序开了多个进程,全局变量也不会共享)
fork函数
- 作用:创建一个子进程
- 返回值:
在父进程中,fork返回新创建子进程的进程PID;
在子进程中,fork返回0;
如果出现错误,fork返回一个负值;
- 实现细节
·OS为子进程创建一个PCB(Process Control Block, 进程控制块,其实质是一个结构体)
·复制代码和数据到子进程的地址空间(PID不复制)
- 获取PID
|
当前进程 |
获取子进程PID |
获取父进程PID |
|
子进程 |
getpid() |
getppid() |
|
父进程 |
pid=fork() |
getpid() |
获取PID代码样例
|
/************************************************************************* > File Name: testfork.c > Author: sinkinben > E-mail: sinkinben@qq.com > Created Time: Sat 23 Mar 2019 08:13:00 AM CST ************************************************************************/
#include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> int main() { //pid_t pid = fork(); pid_t pid; pid = fork(); //sleep(3); //3sec if (pid == 0) printf("In child process: child pid = %d, parent pid = %d\n", getpid(), getppid()); else printf("In parent process: parent pid = %d, child pid = %d\n", getpid(), pid); wait(NULL); return }
/* * 父进程先于子进程退出 * init是1号进程,接管子进程 * 所以in child process, getppid()总是返回1 * 解决办法:父进程调用sleep * 正解:调用wait,让父进程等待子进程结束 */
|
- 并发特性与隔离特性
|
并发 |
·二者并发运行,输出交织。 ·代码段相同:父进程从fork返回处执行(返回值为子进程PID);子进程从fork返回处执行(返回值0) |
|
隔离 |
·地址空间隔离,仅能访问自己的地址空间 ·出现非法访存,只影响当前进程 ·全局变量也不共享 |
并发和隔离样例代码
|
/************************************************************************* > File Name: concurrency.c > Author: sinkinben > E-mail: sinkinben@qq.com > Created Time: Sat 23 Mar 2019 08:38:19 AM CST ************************************************************************/
#include <stdio.h> #include <unistd.h> #include <stdlib.h> static void child() { int i = 0; for (i = 0; i < 5; i++) { puts("child"); printf("global = %d\n", global++); sleep(1); } } void parent() { int i = 0; for (i = 0; i < 5; i++) { puts("parent"); printf("global = %d\n", global++); sleep(1); } }
int main() { pid_t pid = fork(); if (pid) parent(); else child(); return }
|
运行结果

exec系列
名词解析:
exec:execute
l:list,表示参数是列表形式
v:vector,表示参数是数组形式
p:path,允许exec系列函数使用系统环境变量PATH下的路径
全家桶如下。
头文件:unistd.h
返回值:成功无返回值,失败-1.
|
函数 |
原型 |
描述 |
|
execl |
int execl(const char *path, const char *arg, ... |
·当前进程地址空间清空 ·将path指定的可执行程序的代码和数据装入当前地址空间 ·参数可变,最后一个必须NULL ·可以是相对/绝对路径 ·失败原因存于errno(一个Linux全局变量)中,可通过perror()打印。 ·execl成功的话,会执行另外的程序,因此成功无返回值。 ·参数格式:路径+程序名+参数1,2,3...,NULL |
|
execv |
int execv(const char *path, char *const argv[]); |
·argv数组最后一项必须为NULL ·argv[]={程序名,参数1,2,3...,NULL} ·参数格式:路径+argv |
|
execlp |
int execlp(const char *file, const char *arg, ... |
·允许使用PATH环境变量作为参数file |
|
execvp |
int execvp(const char *file, char *const argv[]); |
execl样例代码
|
/************************************************************************* > File Name: testexecl.c > Author: sinkinben > E-mail: sinkinben@qq.com > Created Time: Sat 23 Mar 2019 08:52:03 AM CST ************************************************************************/
#include <stdio.h> #include <unistd.h> int main() { puts("before execl"); int error_code = execl("/bin/echo", "echo", "sin", "cos", NULL); if (error_code < 0) perror("error execl"); puts("after execl"); return }
|
execv样例代码
|
/************************************************************************* > File Name: testexecv.c > Author: sinkinben > E-mail: sinkinben@qq.com > Created Time: Sat 23 Mar 2019 08:59:16 AM CST ************************************************************************/
#include <stdio.h> #include <unistd.h> int main() { puts("before execv(p)"); char *argv[] = { "ls", "-a", NULL}; //int error = execv("/bin/ls",argv); int error = execvp("ls", argv); if(error == -1) perror("error"); puts("after execv(p)"); return
}
|
exit
·头文件:stdlib.h
·原型:void exit(int status);
·功能
>>正常退出当前进程
>>将status & 0XFF作为退出码返回给父进程
·预定义常量
>>EXIT_SUCCESS, 为0,表示正常退出
>>EXIT_FAILURE,为非0,表示异常退出
>>在Linux shell中,$?可以获取程序的退出码
·main与exit
>>return 0 相当于exit(SUCCESS)
>>return !0 相当于exit(FAILURE)
>>start.asm(计算机组成原理知识:main返回的上一层是start.asm)
|
global _start extern main extern exit _start: call main add esp, push eax //作为参数eax->[ebp+8]传给exit call exit add esp, 4 hlt |
exit与$?代码实例
|
#include <stdio.h> #include <stdlib.h> int main() { exit(233); } |
运行结果

atexit
·头文件:stdlib.h
·原型:int atexit(void (*function)(void)); (参数是函数指针)
·功能
>>注册回调函数,进程正常结束时,function被调用
>>如果注册多个,以与注册相反的顺序调用
wait
·头文件:sys/types.h, sys/wait.h
·原型:pid_t wait(int *status);
·功能
>>等待子进程结束
>>如果status非空,子进程退出码保存在status
·退出码
进程可能由于不同原因退出。主动调用exit退出,或接收信号退出。
查询退出原因的宏
|
WIFEXITED(status) |
当子进程exit(0)时,为真。 |
|
WEXITSTATUS(status) |
当子进程exit(!0)时,返回退出码 |
|
WIFSIGNALED(status) |
当子进程接收信号后退出,为真 |
|
WTERMSIG(status) |
当子进程接收信号后退出,返回导致退出的信号码 |
OSLab多进程的更多相关文章
- Python中的多进程与多线程(一)
一.背景 最近在Azkaban的测试工作中,需要在测试环境下模拟线上的调度场景进行稳定性测试.故而重操python旧业,通过python编写脚本来构造类似线上的调度场景.在脚本编写过程中,碰到这样一个 ...
- 取代SharedPreferences的多进程解决方案
Android的SharedPreferences用来存储一些键值对, 但是却不支持跨进程使用. 跨进程来用的话, 当然是放在数据库更可靠啦, 本文主要是给作者的新库PreferencesProvid ...
- python 多进程使用总结
python中的多进程主要使用到 multiprocessing 这个库.这个库在使用 multiprocessing.Manager().Queue时会出问题,建议大家升级到高版本python,如2 ...
- Nginx深入详解之多进程网络模型
一.进程模型 Nginx之所以为广大码农喜爱,除了其高性能外,还有其优雅的系统架构.与Memcached的经典多线程模型相比,Nginx是经典的多进程模型.Nginx启动后以daemon ...
- Python的多线程(threading)与多进程(multiprocessing )
进程:程序的一次执行(程序载入内存,系统分配资源运行).每个进程有自己的内存空间,数据栈等,进程之间可以进行通讯,但是不能共享信息. 线程:所有的线程运行在同一个进程中,共享相同的运行环境.每个独立的 ...
- 进击的Python【第十章】:Python的socket高级应用(多进程,协程与异步)
Python的socket高级应用(多进程,协程与异步)
- PHP的pcntl多进程
PHP使用PCNTL系列的函数也能做到多进程处理一个事务.比如我需要从数据库中获取80w条的数据,再做一系列后续的处理,这个时候,用单进程?你可以等到明年今天了...所以应该使用pcntl函数了. 假 ...
- 初探PHP多进程
h2:first-child, body>h1:first-child, body>h1:first-child+h2, body>h3:first-child, body>h ...
- gdb进程调试,多进程调试
1.单进程的调试 常规的通过gdb cmd这种方式开启调试,特别说明的是通过attach的方法附加到一个指定的进程上去进行调试,这种方法适合于调试一个已经运行的进程,具体用法: gdb -p [pi ...
随机推荐
- 51nod1347 旋转字符串
题目很容易懂,只要进行几次简单的判断就能完成此题,显示判断是否为偶数,之后利用sustr截取两个字符串进行比较,代码如下 #include<iostream> #include<st ...
- 自学java坎坷之路——20155312张竞予
20155312 2006-2007-2 <Java程序设计>第一周学习总结 教材学习内容总结 第一周并没有在课堂上对教材内容进行学习,学习内容概括如下 课程分数构成,其中包括课堂测验(每 ...
- 使用 kbmmw 的ORM开发纯REST数据库访问服务
运行环境: WIN 10 X64 delphi 10.2.2 kbmmw 5.05.11 Firefox 58.0.2 今天使用最新的kbmmw 版本做一个基于ORM的纯数据库访问的REST 服务器 ...
- iOS知识基础篇--@property,@synthesize, nonatomic,atomic,strong,weak,copy,assign,retain详解
一.@property 这个关键词的唯一作用就是声明getter.setter方法接口. 二.@synthesize 实现setter.getter方法,找不到实例变量则主动创建一个. 三.nonat ...
- Tomcat架构解析(一)-----Tomcat总体架构
Tomcat是非常常用的应用服务器,了解Tomcat的总体架构以及实现细节,对于理解整个java web也是有非常大的帮助. 一.Server 1.最简单的服务器结构 最简单的服务器结构如图所示: ...
- (5)How to let go of being a "good" person — and become a better person
https://www.ted.com/talks/dolly_chugh_how_to_let_go_of_being_a_good_person_and_become_a_better_perso ...
- boost--线程同步
1.互斥锁(互斥量) mutex是独占式的互斥锁.timed_mutex增加了超时功能. 成员函数:lock()用于锁定,try_lock()为非阻塞版本的锁定,unlock()用于解锁.timed_ ...
- redis 部分操作
http://pan.baidu.com/s/1eRXdbUy 下载安装地址 打开一个cmd窗口,使用cd命令切换到指定目录(D:\ruanjian\redis\64bit)运行 redis-ser ...
- VMware Authorization Service不能启动 VMware虚拟机状态已挂起无法恢复解决方案
在网上看说在服务里面启动 但也是不能用 电脑上说是WINDOWS无法启动VMware Authorization Service服务(位于本地计算机上)错误:1068 依赖服务或组无法启动 这个很简单 ...
- UVaLive 3641 Leonardo's Notebook (置换)
题意:给定一个置换 B 问是否则存在一个置换 A ,使用 A^2 = B. 析:可以自己画一画,假设 A = (a1, a2, a3)(b1, b2, b3, b4),那么 A^2 = (a1, a2 ...