多個任務同時執行
比如,我們要對已知的用戶數據進行判斷,是否需要發送郵件和短信,如果需要發送則發送。
不使用多進程時,我們首先判斷是否發送郵件,如果需要則發送;然后再判斷是否需要發送短信,如果需要則發送。如果發送郵件耗時2s,發送短信耗時2s,那么我們完成任務大概需要4s左右的時間。
如果我們使用多線程的話,可以開兩個線程,一個用於處理郵件,一個用於處理短信,則耗時一共需要2s左右,處理時間縮短了一半。
以下是代碼實例:
<?php
/**
* Created by PhpStorm.
* User: cyw0413
* Date: 2018/10/20
* Time: 10:37
*/
$info = array(
"sendmail"=>1,
"mailto"=>"12345@qq.com",
"sendsms"=>1,
"smsto"=>"123456"
);
echo "start:".date("Y-m-d H:i:s").PHP_EOL;
$mail_process = new swoole_process('sendMail',true);
$mail_process->start();
$sms_process = new swoole_process('sendSMS',true);
$sms_process->start();
//主進程輸出子進程范圍內容
echo $mail_process->read();
echo PHP_EOL;
echo $sms_process->read();
echo PHP_EOL;
echo "end:".date("Y-m-d H:i:s").PHP_EOL;
//並行函數
function sendMail(swoole_process $worker){
global $info;
if($info['sendmail']==1){
sleep(2);
$worker->write("send mail to ".$info['mailto']);
}
}
function sendSMS(swoole_process $worker){
global $info;
if($info['sendmail']==1){
sleep(2);
$worker->write("send sms to ".$info['smsto']);
}
}
大任務划分成多個小任務
假設我們現在有一個通過curl抓取網頁內容的需求,需要抓取10個網頁,url地址通過數組讀取,每個curl耗時2s。如果我們通過for循環來抓取這10個網頁,需要耗時20s,使用多進程我們可以將任務划分成5份,分別由5個進程執行,每個進程抓取2個url,並發執行,共耗時4s,效率提高5倍。
以下是代碼實例
<?php
/**
* Created by PhpStorm.
* User: cyw0413
* Date: 2018/10/20
* Time: 10:51
*/
$url_arr = array();
for ($i=0;$i<10;$i++){
$url_arr[] = "www.baidu.com?wd=".$i;
}
echo "start:".date("Y-m-d H:i:s").PHP_EOL;
$workers = array();
for ($i=0;$i<5;$i++){
$process = new swoole_process('getContents',true);
$process->start();
$process->write($i);
$workers[] = $process;
}
//主進程數據結果
foreach ($workers as $process){
echo $process->read();
echo PHP_EOL;
}
echo "end:".date("Y-m-d H:i:s").PHP_EOL;
function getContents(swoole_process $worker){
$i = $worker->read();
global $url_arr;
$res1 = execCurl($url_arr[($i*2)]);
$res2 = execCurl($url_arr[($i*2+1)]);
echo $res1.PHP_EOL.$res2;
}
function execCurl($url){
sleep(2);
return "handle ".$url." finished";
}
總結
以上兩種情況,本質上都是將邏輯上沒有先后關系的任務,用多個進程程並發執行,提高效率。
php機制本身不提供多線程的操作,ptcl擴展提供了php操作linux多進程的接口。