easyswoole - 基于swoole扩展实现的一款高性能php框架

協程

協程不是進程或線程,其執行過程更類似于子例程,或者說不帶返回值的函數調用。
一個程序可以包含多個協程,可以對比與一個進程包含多個線程,因而下面我們來比較協程和線程。我們知道多個線程相對獨立,有自己的上下文,切換受系統控制;而協程也相對獨立,有自己的上下文,但是其切換由自己控制,由當前協程切換到其他協程由當前協程來控制。 協程

協程執行順序

首先,我們來看一個原生php代碼:

<?php
function task1(){
    for ($i=0;$i<=300;$i++){
        //寫入文件,大概要3000微秒
        usleep(3000);
        echo "寫入文件{$i}\n";
    }
}
function task2(){
    for ($i=0;$i<=500;$i++){
        //發送郵件給500名會員,大概3000微秒
        usleep(3000);
        echo "發送郵件{$i}\n";
    }
}
function task3(){
    for ($i=0;$i<=100;$i++){
        //模擬插入100條數據,大概3000微秒
        usleep(3000);
        echo "插入數據{$i}\n";
    }
}
task1();
task2();
task3();

在這個代碼中,我們主要做了3件事:寫入文件,發送郵件,以及插入數據. 再看下面這段代碼:

<?php
function task1($i)
{
    //使用$i標識 寫入文件,大概要3000微秒
    if ($i > 300) {
        return false;//超過300不用寫了
    }
    echo "寫入文件{$i}\n";
    usleep(3000);
    return true;
}

function task2($i)
{
    //使用$i標識 發送郵件,大概要3000微秒
    if ($i > 500) {
        return false;//超過500不用發送了
    }
    echo "發送郵件{$i}\n";
    usleep(3000);
    return true;
}

function task3($i)
{
    //使用$i標識 插入數據,大概要3000微秒
    if ($i > 100) {
        return false;//超過100不用插入
    }
    echo "插入數據{$i}\n";
    usleep(3000);
    return true;
}

$i           = 0;
$task1Result = true;
$task2Result = true;
$task3Result = true;
while (true) {
    $task1Result && $task1Result = task1($i);
    $task2Result && $task2Result = task2($i);
    $task3Result && $task3Result = task3($i);
    if($task1Result===false&&$task2Result===false&&$task3Result===false){
        break;//全部任務完成,退出循環
    }
    $i++;
}

這段代碼也是做了3件事,寫入文件,發送郵件,以及插入數據,但是和上面的不同的是,這段代碼將這3件事交叉執行,每個任務執行完一次之后,切換到另一個任務,如此循環.
類似于這樣的執行順序,就是協程.

協程是指一種用代碼實現任務交叉執行的邏輯,協程可以使得代碼1中的3個函數交叉運行,在實現了協程的框架中,我們不需要通過代碼2的方法實現任務交叉執行.直接可讓代碼1中的while(1),執行一次后切換

協程的實現

在php中,實現協程主要使用2種方式:

  • yield生成器實現
  • swoole擴展實現

swoole實現協程代碼:

<?php
function task1(){
    for ($i=0;$i<=300;$i++){
        //寫入文件,大概要3000微秒
        usleep(3000);
        echo "寫入文件{$i}\n";
        Co::sleep(0.001);//掛起當前協程,0.001秒后恢復//相當于切換協程
    }
}
function task2(){
    for ($i=0;$i<=500;$i++){
        //發送郵件給500名會員,大概3000微秒
        usleep(3000);
        echo "發送郵件{$i}\n";
        Co::sleep(0.001);//掛起當前協程,0.001秒后恢復//相當于切換協程
    }
}
function task3(){
    for ($i=0;$i<=100;$i++){
        //模擬插入100條數據,大概3000微秒
        usleep(3000);
        echo "插入數據{$i}\n";
        Co::sleep(0.001);//掛起當前協程,0.001秒后恢復//相當于切換協程
    }
}
$pid1 = go('task1');//go函數是swoole的開啟協程函數,用于開啟一個協程
$pid2 = go('task2');
$pid3 = go('task3');

以上代碼,即可實現切換函數

為什么要用sleep掛起協程實現切換呢?因為swoole的協程是自動的,當協程內遇上I/O操作(mysql,redis)等時,swoole的協程會自動切換,運行到下一個協程任務中(切換后,I/O繼續執行),直到下一個協程任務完成或者被切換(遇上I/O),如此反復,直到所有協程任務完成,則任務完成

協程與進程

由上面的協程執行順序中的代碼2,我們很容易發現,協程其實只是運行在一個進程中的函數,只是這個函數會被切換到下一個執行,可以這么說:

協程只是一串運行在進程中的任務代碼,只是這些任務代碼可以交叉運行 注意,協程并不是多任務并行,屬于多任務串行,每個進程在一個時間只執行了一個任務

協程的作用域

由于協程就是進程中一串任務代碼,所以它的全局變量,靜態變量等變量都是共享的,包括了php的全局緩沖區.
所以,在開發之中,需要特別注意協程中的全局變量,靜態變量,只要某一個協程內修改了,那將會影響全部的協程,在使用ob緩沖區函數攔截的時候,也得考慮是否會被其他協程的輸出給污染.
協程執行順序中的代碼2解釋,當task1給$_GET['name']賦值為1時,task2讀取$_GET['name']也會是1,task2將$_GET['name']賦值為2時,task3讀取$_GET['name']也會是2

協程中的I/O連接

在協程中,要特別注意不能共用一個I/O連接,否則會造成數據異常. 用協程執行順序中的代碼2解釋,當task1,task2函數共用mysql連接,并都進行查詢時,由于協程是交叉運行的,可能會造成task1獲取到task1+task2查詢出來的數據,也可能會丟失部分數據,被task2獲取.

由于協程的交叉運行機制,各個協程的I/O連接都必須是獨立的,所以我們需要在每個協程都創建一個連接,但由于mysql,redis的連接數有限,以及連接的開啟關閉需要消耗大量資源,所以我們可以使用連接池方案實現共用連接(只要保證每個連接每次只有一個協程在使用即可)

主站蜘蛛池模板: 水环式真空泵-旋片式真空泵-上海飞鲁泵业科技有限公司 | 红河州春辉装饰有限公司官网| 首页|YULI玉立专业厨房电器官网|集成灶十大品牌 油烟机 燃气灶 | 腾云网-智慧网络营销服务提供商| 桥梁伸缩缝_桥梁伸缩缝厂家_桥梁伸缩缝价格-衡水淞皓路桥养护工程有限公司 | 铝压铸件_铝合金压铸件_铝合金压铸件厂家-安平县长虹压铸厂 | 真空泵维修保养-真空泵-真空泵油-真空泵过滤器-真空泵碳片-东莞市海扬真空设备有限公司 | 黑料网 - 黑料大事记-黑料门 黑料社最新 今日黑料 热门黑料 最新反差免费-黑料网今日黑料首页_黑料网 - 黑料大事记-黑料门 黑料社最新 今日黑料 热门黑料 最新反差免费-黑料网今日黑料首页 | 上海钧尚电器有限公司 - Faulhaber电机 AMETEK pittman电机 AMETEK ROTRON军用航空风机 Exlar电动缸 MAE电机 MCG电机 CP电动工具 马头工具 AMCI驱动器 直流电机 减速箱 直流伺服电机,无刷电机,直线电机 直流防爆电机 防爆电机 汽车助力转向电机 EPS电机 faulhaber motor faulhaber gearbox NANOTEC电机 ELWOOD电机 PHYTRON电机 EXLAR伺服电动缸 高力矩、高性能直流电机,音圈电机,风机,直流风机,航空风机 | 消防服_防化服_灭火防护服_消防装备生产厂家-泰州市华通消防装备厂有限公司 | 重庆污水处理设备_废气处理设备_纯净水设备-山艺环保 | 上海山田机械有限公司| 瑞安网站建设_网站制作_做网站找瑞安联创网络:专业网页制作、手机网站、SEO百度网站关键字优化 软文营销推广-新闻稿发布-软文撰写-百科词条编辑-品牌全案策划推广网络营销传播-喜尚传媒 | 溶气气浮机_一体化净水设备_污水处理设备_mbr一体化污水处理设备-明基环保 | 湖南长沙癫痫病医院_湖南癫痫病专科医院_湖南治疗癫痫病专业医院_长沙和谐医院有限公司 | 内蒙古天奇生物科技有限公司——OEM代工|ODM定制|原料供应|骨肽|片剂|胶囊剂|口服液 | 紫外交联仪,紫外透射仪,紫外灯-上海析浦科学仪器有限公司 | 鸿茗商务-杭州鸿茗商务咨询有限公司 | 上海拓展训练_上海拓展培训_上海团建活动_上海团建策划 | 西安外墙防水补漏-堵漏-防腐-保温工程公司-陕西宏图祥瑞实业有限公司 | 透明捆扎带_束带机打包带_束带机纸带_热封纸带机_上海得亿束带机包装材料有限公司 | 消防施工,消防工程施工,消防施工改造-北京消防工程公司-亿杰(北京)消防工程有限公司 | 久久91精品久久91综合_国产亚洲自拍一区_国产精品第1页_亚洲高清视频一区_91成人午夜在线精品_亚洲国产精品网站在线播放_亚洲国产成人久久综合区_国产精品亚洲专区在线观看_免费视频精品一区二区三区 | 上海国际餐饮博览会|餐饮供应|餐饮服务|餐饮加盟 | 上海钧尚电器有限公司 - Faulhaber电机 AMETEK pittman电机 AMETEK ROTRON军用航空风机 Exlar电动缸 MAE电机 MCG电机 CP电动工具 马头工具 AMCI驱动器 直流电机 减速箱 直流伺服电机,无刷电机,直线电机 直流防爆电机 防爆电机 汽车助力转向电机 EPS电机 faulhaber motor faulhaber gearbox NANOTEC电机 ELWOOD电机 PHYTRON电机 EXLAR伺服电动缸 高力矩、高性能直流电机,音圈电机,风机,直流风机,航空风机 | 鑫金牛建设工程(苏州)有限公司 | 意大利留学-意大利语培训-马来西亚留学【长青藤海外】 | 河北高新技术企业认定,沧州商标注册,沧州9001质量管理体系认证,沧州高新技术企业认定,沧州体系认证,沧州商标续展,沧州版权登记,河北国瑞企业管理咨询有限公司 | 曙海培训-仿真培训Linux培训html5培单片机培训PCB培训python培训PLC培训C语言培训android培训物联网培训无线电培训欧姆龙培训工业机器人培训5G培训Hadoop培训CFD培训项目外包开发咨询 | 液压升降货梯_导轨式升降机_往复式提升机_济南宇轩机械厂家 | 液压油缸生产厂家-常州辰佰液压机械有限公司 | 行李快递安检机-便携式X光安检仪-行李安检机-液体-爆炸物探测仪-安天下安检设备 | 乌海市腐植酸盐加工,内蒙古腐植酸钠,内蒙古型煤粘合剂生产厂家|创联腐植酸盐加工有限公司 | 树脂门|卫浴门|医院门|医院专用门|卫生间门—湖南航天康达树脂门 | 山东亮化工程_亮化公司_亮化资质-山东星汇照明工程有限公司 | 轻型防化服|重型防化服|全封闭防化服|济南三安安全防护设备有限公司 | 原创软文新闻稿-网站SEO文章代写-征文演讲稿代笔-写作阁 | 雨量计,翻斗雨量计,声波雨量计i,时差法流量计,时差法明渠流量计 - 徐州伟思水务科技有限公司 | 锯轨机|气动锯轨机|内燃锯轨机|电动钢轨钻孔机|内燃钢轨钻孔机-山东卓力铁路设备有限公司 | 四方光电(武汉)仪器有限公司_四方仪器首页-烟气分析仪|尾气分析仪|煤气分析仪|沼气分析仪|天然气分析仪|超声波流量计|在线气体分析系统|红外气体传感器 | 小型生活污水处理设备_MBR膜生物反应器_口腔医院/脱脂污水处理设备_酸洗磷化/喷涂废水处理设备-上海台江环保 |