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里进程创建和分析的更多相关文章

  1. Linux内核分析--进程创建,执行,切换

    学号:351 原创作品转载请注明出处本实验来源 https://github.com/mengning/linuxkernel/ 实验要求 从整理上理解进程创建.可执行文件的加载和进程执行进程切换,重 ...

  2. 从整体上理解进程创建、可执行文件的加载和进程执行进程切换,重点理解分析fork、execve和进程切换

    学号后三位<168> 原创作品转载请注明出处https://github.com/mengning/linuxkernel/ 1.分析fork函数对应的内核处理过程sys_clone,理解 ...

  3. Windows进程创建的流程分析

    .   创建进程的大体流程:   创建进程的过程就是构建一个环境,这个环境包含了很多的机制 (比如自我保护, 与外界通信等等). 构建这个环境需要两种"人"来协调完成(用户态和内核 ...

  4. Android虚拟机器学习总结Dalvik虚拟机创建进程和线程分析

    Dalvik调用一个成员函数时,虚拟机,假设发现,该成员函数是一个JNI办法,然后,它会直接跳转到其地址来运行.也就是说.JNI方法是直接在本地操作系统上运行的.而不是由Dalvik虚拟机解释器运行. ...

  5. Linux-进程描述(4)之进程优先级与进程创建执行

    进程优先级 进程cpu资源分配就是指进程的优先权(priority).优先权高的进程有优先执行权利. 权限与优先级.权限(privilege)是指在多用户计算机系统的管理中,某个特定的用户具有特定的系 ...

  6. python的multiprocessing模块进程创建、资源回收-Process,Pool

    python的multiprocessing有两种创建进程的方式,每种创建方式和进程资源的回收都不太相同,下面分别针对Process,Pool及系统自带的fork三种进程分析. 1.方式一:fork( ...

  7. Linux学习--进程创建

    进程创建 在Linux系统下,自己可以创建进程: 当进程执行时,它会被装载进虚拟内存,为程序变量分配空间,并把相关信息添到 task_struct里. 进程内存布局分为四个不同的段: • 文本段,包含 ...

  8. 进程创建过程详解 CreateProcess

    转载请您注明出处:http://www.cnblogs.com/lsh123/p/7405796.html 0x01 CreateProcessW CreateProcess的使用有ANSI版本的Cr ...

  9. 理解Android进程创建流程(转)

    /frameworks/base/core/java/com/android/internal/os/ - ZygoteInit.java - ZygoteConnection.java - Runt ...

随机推荐

  1. Android(Lollipop/5.0) Material Design(二) 入门指南

    Material Design系列 Android(Lollipop/5.0)Material Design(一) 简介 Android(Lollipop/5.0)Material Design(二) ...

  2. Dynamics CRM EntityCollection 根据实体中的某个字段为依据去除重复数据

    CRM中通过QueryExpression查询出了一个EntityCollection集,但有时会存在重复数据,QueryExpression中有个属性distinct,只要设置为true就能过滤 ...

  3. ISLR系列:(3)重采样方法 Cross-Validation & Bootstrap

    Resampling Methods 此博文是 An Introduction to Statistical Learning with Applications in R 的系列读书笔记,作为本人的 ...

  4. 浅谈SystemClock 和Thead的区别和联系

    其实将SystemClock 和Thead直接放在一起是不合适的,我们首先来看下他们所在的api. public final class SystemClock extends Object java ...

  5. java反射案例详解

    白首为功名.旧山松竹老,阻归程.欲将心事付瑶琴.知音少,弦断有谁听? [案例1]通过一个对象获得完整的包名和类名 package Reflect; /** * 通过一个对象获得完整的包名和类名 * * ...

  6. 一个 developer 的进化

    作为一名开发者已十年,回顾过往大概经历了这么几个阶段,如下图所示: Develop Code 作为刚走出学校的学生进入公司,在最初的 1-2 年内就处于该阶段. 不停的开发代码,为系统的大厦添砖加瓦, ...

  7. 套接字编程相关函数(2:TCP套接字编程相关函数)

    本文摘录自<UNIX网络编程 卷1>. 基本套接字函数 socket函数 为了执行网络I/O,一个进程必须做的第一件事就是调用socket函数,指定期望的通信协议类型.其定义如下: #in ...

  8. OV5640全景模式预览倒180度,拍照正常的问题

    此方法基本上适用于所有android平台上全景模式预览倒180度,拍照正常的问题. 首先说明的是,影响camera方向的有三个地方,分别是系统方向,内核camera方向和驱动镜像.全景模式预览只跟系统 ...

  9. C++ 传参时传内置类型时用传值(pass by value)方式效率较高

    来源:唐磊的个人博客<C++ 传参时传内置类型时用传值(pass by value)方式效率较高> 在<Effective C++>里提到对内置(C-like)类型在函数传参时 ...

  10. OpenCV 实现分水岭算法

    种子点的标记没有太搞懂,这个算法的速度还是很快的 // watershed_test20140801.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h&q ...