## php的多進程實驗分析
>[tip] 由于時間久遠,本文不再具有參考性,請參閱更為易讀新文章:[進程 · php筆記 · 看云](http://www.hmoore.net/xiak/php-node/785180)
pcntl_fork — 在當前進程當前位置產生分支(子進程)。譯注:fork是創建了一個子進程,父進程和子進程 都從fork的位置開始向下繼續執行,不同的是父進程執行過程中,得到的fork返回值為子進程 號,而子進程得到的是0。
```php
<?php
fwrite(STDOUT, "stat>" . PHP_EOL.PHP_EOL);
$count = 3;
for ($i = 0; $i < $count; $i++) {
$pid = pcntl_fork();
if($pid == -1) {
fwrite(STDOUT, "Could not fork worker {$i}" . PHP_EOL);
die();
}
else if(!$pid) {
fun($i, $count);
// 如果不使用下面的方式退出,將會出現什么呢,經測試,下面是我們得出的結果。
// break;
// exit;
} else {
fwrite(STDOUT, "父進程 " . $i . PHP_EOL);
}
}
function fun($i, $count)
{
// global $num;
$num = $i;
$jc = exec('echo $$');
$time = date('Y-m-d H:i:s');
fwrite(STDOUT, "進程:$jc 計數:$num - $count 時間:$time" . PHP_EOL);
fwrite(STDOUT, "start > " . date('Y-m-d H:i:s') . PHP_EOL);
fwrite(STDOUT, "~ 3s ~" . PHP_EOL);
sleep(3);
fwrite(STDOUT, "enddd $i < " . date('Y-m-d H:i:s') . PHP_EOL.PHP_EOL);
}
```
父進程循環 從0開始 3次 創建3個子進程 產生下面第一部分
i = 0/1/2 結束循環
子0進程下面的循環 從1開始循環 能循環2次 創建2個子進程 產生下面第二部分(由第一子進程產生) 和 第三部分(由第二子進程產生)
子1進程下面的循環 從2開始循環 能循環1次 創建1個子進程 產生下面第四部分
子2進程下面的循環 沒有循環 退出了(什么都沒做)
由此分析可見PHP的這個子進程機制的特點:
繼承父進程的“運行上下文”即變量和環境都是直接繼承了,并且各個子進程相互,不受影響。
~~~
[root@iZ28yn5lehbZ x]# x=sf php d3.php
stat>
父進程 0
父進程 1
父進程 2
進程:25608 計數:0 - 3 時間:2016-08-15 08:35:47
start > 2016-08-15 08:35:47
~ 3s ~
進程:25610 計數:2 - 3 時間:2016-08-15 08:35:47
start > 2016-08-15 08:35:47
~ 3s ~
進程:25611 計數:1 - 3 時間:2016-08-15 08:35:47
start > 2016-08-15 08:35:47
~ 3s ~
enddd 1 < 2016-08-15 08:35:50
enddd 2 < 2016-08-15 08:35:53
enddd 0 < 2016-08-15 08:35:50
父進程 1
父進程 2
進程:25643 計數:1 - 3 時間:2016-08-15 08:35:50
start > 2016-08-15 08:35:50
~ 3s ~
進程:25644 計數:2 - 3 時間:2016-08-15 08:35:50
start > 2016-08-15 08:35:50
~ 3s ~
enddd 1 < 2016-08-15 08:35:53
enddd 2 < 2016-08-15 08:35:53
父進程 2
進程:25646 計數:2 - 3 時間:2016-08-15 08:35:50
start > 2016-08-15 08:35:50
~ 3s ~
enddd 2 < 2016-08-15 08:35:50
父進程 2
進程:25663 計數:2 - 3 時間:2016-08-15 08:35:53
start > 2016-08-15 08:35:53
~ 3s ~
enddd 2 < 2016-08-15 08:35:56
~~~
php獲取子進程ID
```
$jc = exec('echo $$'); // 這方式獲取的不是當前PHP腳本執行的進程ID
$child_id = getmypid(); // 這個才是當前PHP腳本執行的進程ID
```
參見:https://segmentfault.com/q/1010000004634861
```
父進程:17488 字進程:17489 計數:0
父進程:17488 字進程:17490 計數:1
進程:17489 計數:0 - 3 時間:2016-08-15 16:55:22
start > 2016-08-15 16:55:22
~ 3s ~
父進程:17488 字進程:17491 計數:2
進程:17490 計數:1 - 3 時間:2016-08-15 16:55:22
start > 2016-08-15 16:55:22
~ 3s ~
進程:17491 計數:2 - 3 時間:2016-08-15 16:55:22
start > 2016-08-15 16:55:22
~ 3s ~
[root@iZ28yn5lehbZ x]# enddd 0 < 2016-08-15 16:55:25
enddd 1 < 2016-08-15 16:55:25
enddd 2 < 2016-08-15 16:55:25
```
上面這個父進程執行部分是否先輸出來,有時并無實際規律,理論上和實際上不一樣,原因是可能需要考慮進程在CPU資源分配使用上面。(但好像父進程 計數:0永遠在第一,也就是說在這個開辟子進程的結構中,父進程是首次執行的。)
```
$status = '';
$pid = pcntl_fork();
//父進程和子進程都會執行下面代碼
if ($pid == -1) {
//錯誤處理:創建子進程失敗時返回-1.
die('could not fork');
} else if ($pid) {
//父進程會得到子進程號,所以這里是父進程執行的邏輯
// pcntl_wait($status); //等待子進程中斷,防止子進程成為僵尸進程。
fwrite(STDOUT, "父進程" . PHP_EOL);
} else {
//子進程得到的$pid為0, 所以這里是子進程執行的邏輯。
fwrite(STDOUT, "子進程" . PHP_EOL);
}
```
**分析**
1. 父進程執行完畢就退出了,但是這個退出不會結束可能還沒執行完成的子進程,也就是說子進程會變成僵尸進程,這并不像結束了ssh就結束了bash,就結束了我們的腳本那樣。(這個也確實是要成為僵尸進程,不然孩子剛生出來還沒玩夠就要死了)
2. 輸出順序為:
父進程
子進程
如果將pcntl_wait這一行注釋掉,輸出順序就為
子進程
父進程
**測試:**
父進程和子進程 都從fork的位置開始向下繼續執行,所以這個硬要說規律也不好說,這個是CPU指令執行資源的事了。

* * * * *
### 擴展
[從 0 到 1 優雅的實現PHP多進程管理](https://mp.weixin.qq.com/s/lbGCQu7zkKUfPhFMFbHooQ)
[TIGERB/naruto: An object-oriented multi process manager for PHP](https://github.com/TIGERB/naruto)
* * * * *
last update:2017-12-18 14:55:26
- 開始
- 公益
- 更好的使用看云
- 推薦書單
- 優秀資源整理
- 技術文章寫作規范
- SublimeText - 編碼利器
- PSR-0/PSR-4命名標準
- php的多進程實驗分析
- 高級PHP
- 進程
- 信號
- 事件
- IO模型
- 同步、異步
- socket
- Swoole
- PHP擴展
- Composer
- easyswoole
- php多線程
- 守護程序
- 文件鎖
- s-socket
- aphp
- 隊列&并發
- 隊列
- 講個故事
- 如何最大效率的問題
- 訪問式的web服務(一)
- 訪問式的web服務(二)
- 請求
- 瀏覽器訪問阻塞問題
- Swoole
- 你必須理解的計算機核心概念 - 碼農翻身
- CPU阿甘 - 碼農翻身
- 異步通知,那我要怎么通知你啊?
- 實時操作系統
- 深入實時 Linux
- Redis 實現隊列
- redis與隊列
- 定時-時鐘-阻塞
- 計算機的生命
- 多進程/多線程
- 進程通信
- 拜占庭將軍問題深入探討
- JAVA CAS原理深度分析
- 隊列的思考
- 走進并發的世界
- 鎖
- 事務筆記
- 并發問題帶來的后果
- 為什么說樂觀鎖是安全的
- 內存鎖與內存事務 - 劉小兵2014
- 加鎖還是不加鎖,這是一個問題 - 碼農翻身
- 編程世界的那把鎖 - 碼農翻身
- 如何保證萬無一失
- 傳統事務與柔性事務
- 大白話搞懂什么是同步/異步/阻塞/非阻塞
- redis實現鎖
- 淺談mysql事務
- PHP異常
- php錯誤
- 文件加載
- 路由與偽靜態
- URL模式之分析
- 字符串處理
- 正則表達式
- 數組合并與+
- 文件上傳
- 常用驗證與過濾
- 記錄
- 趣圖
- foreach需要注意的問題
- Discuz!筆記
- 程序設計思維
- 抽象與具體
- 配置
- 關于如何學習的思考
- 編程思維
- 談編程
- 如何安全的修改對象
- 臨時
- 臨時筆記
- 透過問題看本質
- 程序后門
- 邊界檢查
- session
- 安全
- 王垠
- 第三方數據接口
- 驗證碼問題
- 還是少不了虛擬機
- 程序員如何談戀愛
- 程序員為什么要一直改BUG,為什么不能一次性把代碼寫好?
- 碎碎念
- 算法
- 實用代碼
- 相對私密與絕對私密
- 學習目標
- 隨記
- 編程小知識
- foo
- 落盤
- URL編碼的思考
- 字符編碼
- Elasticsearch
- TCP-IP協議
- 碎碎念2
- Grafana
- EFK、ELK
- RPC
- 依賴注入
- 科目一
- 開發筆記
- 經緯度格式轉換
- php時區問題
- 解決本地開發時調用遠程AIP跨域問題
- 后期靜態綁定
- 談tp的跳轉提示頁面
- 無限分類問題
- 生成微縮圖
- MVC名詞
- MVC架構
- 也許模塊不是唯一的答案
- 哈希算法
- 開發后臺
- 軟件設計架構
- mysql表字段設計
- 上傳表如何設計
- 二開心得
- awesomes-tables
- 安全的代碼部署
- 微信開發筆記
- 賬戶授權相關
- 小程序獲取是否關注其公眾號
- 支付相關
- 提交訂單
- 微信支付筆記
- 支付接口筆記
- 支付中心開發
- 下單與支付
- 支付流程設計
- 訂單與支付設計
- 敏感操作驗證
- 排序設計
- 代碼的運行環境
- 搜索關鍵字的顯示處理
- 接口異步更新ip信息
- 圖片處理
- 項目搭建
- 閱讀文檔的新方式
- mysql_insert_id并發問題思考
- 行鎖注意事項
- 細節注意
- 如何處理用戶的輸入
- 不可見的字符
- 抽獎
- 時間處理
- 應用開發實戰
- python 學習記錄
- Scrapy 教程
- Playwright 教程
- stealth.min.js
- Selenium 教程
- requests 教程
- pyautogui 教程
- Flask 教程
- PyInstaller 教程
- 蜘蛛
- python 文檔相似度驗證
- thinkphp5.0數據庫與模型的研究
- workerman進程管理
- workerman網絡分析
- java學習記錄
- docker
- 筆記
- kubernetes
- Kubernetes
- PaddlePaddle
- composer
- oneinstack
- 人工智能 AI
- 京東
- pc_detailpage_wareBusiness
- doc
- 電商網站設計
- iwebshop
- 商品規格分析
- 商品屬性分析
- tpshop
- 商品規格分析
- 商品屬性分析
- 電商表設計
- 設計記錄
- 優惠券
- 生成唯一訂單號
- 購物車技術
- 分類與類型
- 微信登錄與綁定
- 京東到家庫存系統架構設計
- crmeb
- 命名規范
- Nginx https配置
- 關于人工智能
- 從人的思考方式到二叉樹
- 架構
- 今日有感
- 文章保存
- 安全背后: 瀏覽器是如何校驗證書的
- 避不開的分布式事務
- devops自動化運維、部署、測試的最后一公里 —— ApiFox 云時代的接口管理工具
- 找到自己今生要做的事
- 自動化生活
- 開源與漿果
- Apifox: API 接口自動化測試指南