php里进程创建和分析
Linux系统中,进程的执行模式划分为用户模式和内核模式,当进程运行于用户空间时属于用户模式,
如果在用户程序运行过程中出现系统调用或者发生中断事件,就要运行操作系统(即核心)程序,进程的运行模式就变为内核模式
在该模式下运行的进程可以执行机器特权指令,而且该进程的运行不受用户的干预。
pcntl_fork()函数创建一个子进程,这个子进程仅PID(进程号) 和PPID(父进程号)与其父进程不同
成功时,在父进程执行线程内返回产生的子进程的PID,在子进程执行线程内返回0。失败时,在 父进程上下文返回-1,不会创建子进程调用函数创建进程的时候,函数执行是有时间的,而新的进程刚好是在函数执行开始和结束之间创建出来的,
这样,新的进程也执行了这个函数,所以函数也需要有返回值。
那么对于该函数一次执行之后,父进程和子进程都会受到该函数的返回值
由于父进程创建了子进程,而子进程并没有创建新的进程,所以子进程对于这个函数的返回结果是没有的,
所以就给他赋了一个0
而父进程创建了子进程,子进程是存在pid的,所以就得到了那个进程的pid
下面不考虑进程创建失败的情况
<?php
echo "###\n";
echo posix_getpid();
echo "\n------------\n";
$i=0;
while($i<2){
$pid = pcntl_fork();
echo "now pid:".sprintf("%5d",posix_getpid())."--->父pid:".sprintf("%5d",posix_getppid())."--->fork_return:".sprintf("%5d",$pid)."----->p_".$i++.PHP_EOL;
}
exit;
输出和简析
###
15280
------------
now pid:15280--->父pid:14413--->fork_return:15281----->p_0
now pid:15281--->父pid:15280--->fork_return: 0----->p_0
now pid:15280--->父pid:14413--->fork_return:15282----->p_1
now pid:15282--->父pid:15280--->fork_return: 0----->p_1
now pid:15281--->父pid:15280--->fork_return:15283----->p_1
now pid:15283--->父pid:15281--->fork_return: 0----->p_1
发现打印了4次p_1
分析
当前pid 15280
i=0时记为step1
15280分裂成15280(pcntl_fork()返回15281)
15280分裂成15281(pcntl_fork()返回0)
i=1时记为step2
step1的15280分裂成15280(pcntl_fork()返回15282)
step1的15280分裂成15282(pcntl_fork()返回0) step1的15281分裂成15281(pcntl_fork()返回15283)
step1的15281分裂成15283(pcntl_fork()返回0)
返回0的表示创建了新的子进程在子进程的返回
现在我们只要2个进程(更多也原理一样)
如果使用循环
比如我们要5个进程
<?php
echo "###\n";
echo posix_getpid();
echo "\n------------\n";
$i=0;
while($i<5){
$pid = pcntl_fork();
echo "now pid:".sprintf("%5d",posix_getpid())."--->父pid:".sprintf("%5d",posix_getppid())."--->fork_return:".sprintf("%5d",$pid)."----->p_".$i++.PHP_EOL;
//子进程里的返回,不再进行while的pcntl_fork
if($pid == 0){
echo posix_getpid()." is a child procress!\n";
return '';
}
}
exit;
?>
输出
###
15262
------------
now pid:15262--->父pid:14413--->fork_return:15263----->p_0
now pid:15263--->父pid:15262--->fork_return: 0----->p_0
15263 is a child procress!
now pid:15262--->父pid:14413--->fork_return:15264----->p_1
now pid:15264--->父pid:15262--->fork_return: 0----->p_1
15264 is a child procress!
now pid:15262--->父pid:14413--->fork_return:15265----->p_2
now pid:15265--->父pid:15262--->fork_return: 0----->p_2
15265 is a child procress!
now pid:15262--->父pid:14413--->fork_return:15266----->p_3
now pid:15262--->父pid:14413--->fork_return:15267----->p_4
now pid:15267--->父pid:15262--->fork_return: 0----->p_4
15267 is a child procress!
now pid:15266--->父pid:15262--->fork_return: 0----->p_3
15266 is a child procress!
这时由 15262产生了15263,15264,15265,15266,15267
可以看出所有子进程都在15262里fork出的,是15262的直接子进程
php里进程创建和分析的更多相关文章
- Linux内核分析--进程创建,执行,切换
学号:351 原创作品转载请注明出处本实验来源 https://github.com/mengning/linuxkernel/ 实验要求 从整理上理解进程创建.可执行文件的加载和进程执行进程切换,重 ...
- 从整体上理解进程创建、可执行文件的加载和进程执行进程切换,重点理解分析fork、execve和进程切换
学号后三位<168> 原创作品转载请注明出处https://github.com/mengning/linuxkernel/ 1.分析fork函数对应的内核处理过程sys_clone,理解 ...
- Windows进程创建的流程分析
. 创建进程的大体流程: 创建进程的过程就是构建一个环境,这个环境包含了很多的机制 (比如自我保护, 与外界通信等等). 构建这个环境需要两种"人"来协调完成(用户态和内核 ...
- Android虚拟机器学习总结Dalvik虚拟机创建进程和线程分析
Dalvik调用一个成员函数时,虚拟机,假设发现,该成员函数是一个JNI办法,然后,它会直接跳转到其地址来运行.也就是说.JNI方法是直接在本地操作系统上运行的.而不是由Dalvik虚拟机解释器运行. ...
- Linux-进程描述(4)之进程优先级与进程创建执行
进程优先级 进程cpu资源分配就是指进程的优先权(priority).优先权高的进程有优先执行权利. 权限与优先级.权限(privilege)是指在多用户计算机系统的管理中,某个特定的用户具有特定的系 ...
- python的multiprocessing模块进程创建、资源回收-Process,Pool
python的multiprocessing有两种创建进程的方式,每种创建方式和进程资源的回收都不太相同,下面分别针对Process,Pool及系统自带的fork三种进程分析. 1.方式一:fork( ...
- Linux学习--进程创建
进程创建 在Linux系统下,自己可以创建进程: 当进程执行时,它会被装载进虚拟内存,为程序变量分配空间,并把相关信息添到 task_struct里. 进程内存布局分为四个不同的段: • 文本段,包含 ...
- 进程创建过程详解 CreateProcess
转载请您注明出处:http://www.cnblogs.com/lsh123/p/7405796.html 0x01 CreateProcessW CreateProcess的使用有ANSI版本的Cr ...
- 理解Android进程创建流程(转)
/frameworks/base/core/java/com/android/internal/os/ - ZygoteInit.java - ZygoteConnection.java - Runt ...
随机推荐
- 批量替换数据库中所有用户数据表中字段数据类型为char和varchar到nvarchar的脚本
解决问题:字段类型为char的总是占用指定字节长度(末尾好多空白符号),varchar数据类型长度一个汉字占2个字节,内容存储为中文的字段个人建议全部使用nvarchar. 操作说明:打开SQL Se ...
- android官方技术文档翻译——Case 标签中的常量字段
本文译自androd官方技术文档<Non-constant Fields in Case Labels>,原文地址:http://tools.android.com/tips/non-co ...
- 9.6、Libgdx之罗盘
(官网:www.libgdx.cn) 有些Android和iOS设备可能需要检测使用罗盘检测方向. 注意:罗盘当前在iOS设备中不可用,RoboVM暂不支持. 查询当前罗盘当前是否可用: boolea ...
- 学习tornado:介绍
前言 在python里面,有许多web framework.对于我来说,因为很长一段时间都在使用tornado,所以有了一些心得体会.虽然现在我的工作重点已经转向go了,但是我觉得还是有必要记录一下t ...
- JavaScript进阶(三)常见工具(校验、通用)
JS常见工具(校验.通用) // 姓名校验 var checkName = function(name) { // 收货人姓名校验(准则:姓名为2-4汉字) var regu = /^[\u4E00- ...
- libevent之event_base
event_base是libevent的事务处理框架,负责事件注册.删除等,属于Reactor模式中的Reactor. event_base结构体 event_base结构体定义于<event_ ...
- 【翻译】Siesta事件记录器入门
原文:Getting started with the Siesta event recorder 作者:Mats Bryntse 随着事件记录器功能的发布越来越近,我们准备了一下入门指南,向大家展示 ...
- InfiniDB 修改一行的效率?
InfiniDB引擎的DML速度比较慢,无论设置自动提交开关为关闭或开启,插入性能都很糟糕,但更新和删除的效率还可以,并且不支持truncate表操作. 删,改 效率高 插入,效率低(测试,在数据量稍 ...
- MTK Android添加驱动模块
1 [编写linux驱动程序] 1.1 一.编写驱动核心程序 1.2 二.配置Kconfig 1.3 三.配置Makefile 1.4 四.配置系统的autoconfig 1.5 五.编译 2 [编写 ...
- iOS 网络编程模式总结
IOS 可以采用三类api 接口进行网络编程,根据抽象层次从低到高分别为socket方式.stream方式.url 方式. 一 .socket 方式 IOS 提供的socket 方式的网络编程接口为C ...