成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

關(guān)于PHP協(xié)程與阻塞的思考

FullStackDeveloper / 3158人閱讀

摘要:線程擁有自己獨(dú)立的棧和共享的堆,共享堆,不共享?xiàng)?,線程亦由操作系統(tǒng)調(diào)度標(biāo)準(zhǔn)線程是的。以及鳥哥翻譯的這篇詳細(xì)文檔我就以他實(shí)現(xiàn)的協(xié)程多任務(wù)調(diào)度為基礎(chǔ)做一下例子說明并說一下關(guān)于我在阻塞方面所做的一些思考。

進(jìn)程、線程、協(xié)程

關(guān)于進(jìn)程、線程、協(xié)程,有非常詳細(xì)和豐富的博客或者學(xué)習(xí)資源,我不在此做贅述,我大致在此介紹一下這幾個(gè)東西。

進(jìn)程擁有自己獨(dú)立的堆和棧,既不共享堆,亦不共享?xiàng)?,進(jìn)程由操作系統(tǒng)調(diào)度。

線程擁有自己獨(dú)立的棧和共享的堆,共享堆,不共享?xiàng)?,線程亦由操作系統(tǒng)調(diào)度(標(biāo)準(zhǔn)線程是的)。

協(xié)程和線程一樣共享堆,不共享?xiàng)?,協(xié)程由程序員在協(xié)程的代碼里顯示調(diào)度。

PHP中的協(xié)程實(shí)現(xiàn)基礎(chǔ) yield

yield的根本實(shí)現(xiàn)是生成器類,而迭代器類是迭代器接口的實(shí)現(xiàn):

Generator?implements?Iterator?{
    public?mixed?current?(?void?) // 返回當(dāng)前產(chǎn)生的值
    public?mixed?key?(?void?) // 返回當(dāng)前產(chǎn)生的鍵
    public?void?next?(?void?) // 生成器繼續(xù)執(zhí)行
    public?void?rewind?(?void?) // 重置迭代器,如果迭代已經(jīng)開始了,這里會(huì)拋出一個(gè)異常。
                                             // renwind的執(zhí)行將會(huì)導(dǎo)致第一個(gè)yield被執(zhí)行, 并且忽略了他的返回值.
    public?mixed?send?(?mixed?$value?) // 向生成器中傳入一個(gè)值,并且當(dāng)做?yield?表達(dá)式的結(jié)果,然后繼續(xù)執(zhí)行生成器。如果當(dāng)這個(gè)方法被調(diào)用時(shí),生成器   
                                            // 不在?yield?表達(dá)式,那么在傳入值之前,它會(huì)先運(yùn)行到第一個(gè)?yield?表達(dá)式。
    public?void?throw?(?Exception?$exception?) // 向生成器中拋入一個(gè)異常
    public?bool?valid?(?void?) // 檢查迭代器是否被關(guān)閉
    public?void?__wakeup?(?void?) // 序列化回調(diào),拋出一個(gè)異常以表示生成器不能被序列化。
}

以上解析可以參考PHP官方文檔。

http://php.net/manual/zh/clas...

以及鳥哥翻譯的這篇詳細(xì)文檔:

http://www.laruence.com/2015/...

我就以他實(shí)現(xiàn)的協(xié)程多任務(wù)調(diào)度為基礎(chǔ)做一下例子說明并說一下關(guān)于我在阻塞方面所做的一些思考。

自定義簡(jiǎn)單定時(shí)執(zhí)行任務(wù)示例:

(此例子必須依賴于以上鳥哥實(shí)現(xiàn)的協(xié)程調(diào)度代碼)


class timer {
    private $start = 0; // 定時(shí)開始時(shí)間
    private $timer; // 間隔的時(shí)間差,單位秒
    private $value = 0; // 產(chǎn)生的結(jié)果值
    private $callback; // 異步回調(diào)
    private $isEnd = false; // 當(dāng)前定時(shí)器任務(wù)是否結(jié)束
    public function __construct($timer,callable $callback)
    {
        $this->start = time();
        $this->timer = $timer;
        $this->callback = $callback;
    }
    public function run() {
        if($this->valid()) {
            $callback = $this->callback;
            $callback($this->value ++,$this);
            $this->start = time();
        }
    }
    /**
     * 定時(shí)執(zhí)行檢查
     */
    public function valid() {
        $end = time();
        if($end - $this->start >= $this->timer) {
            return true;
        } else {
            return false;
        }
    }
    public function setEnd($isEnd) {
        $this->isEnd = $isEnd;
    }
    public function getEnd() {
        return $this->isEnd;
    }
}

/**
 * 模擬阻塞的協(xié)程1
 *
 */
function taskObject1() {
    $timer = new timer(1,function($value,timer $timer) {
        if($value >= 5) {
            $timer->setEnd(true);
        }
        echo "
"."A ".$value; }); $tid = (yield getTaskId()); while (true) { if($timer->getEnd() == true) { break; } yield $timer->run(); } } /** * 模擬阻塞的協(xié)程2 * */ function taskObject2() { $timer = new timer(2,function($value,timer $timer) { if($value >= 3) { $timer->setEnd(true); } echo "
"."B ".$value; }); $tid = (yield getTaskId()); while (true) { if($timer->getEnd() == true) { break; } yield $timer->run(); } } $scheduler = new Scheduler; $scheduler->newTask(taskObject1()); $scheduler->newTask(taskObject2()); $scheduler->run();

以上實(shí)現(xiàn)的是:

產(chǎn)生兩個(gè)任務(wù),并行執(zhí)行,并且給每個(gè)任務(wù)在執(zhí)行的時(shí)候模擬幾秒鐘的阻塞;

讓協(xié)程切換的時(shí)候能順利切換,其中的任務(wù)阻塞不相互影響;

思考:

我為什么要做以上這件事情呢?因?yàn)槲野l(fā)現(xiàn)協(xié)程實(shí)現(xiàn)雖然很強(qiáng)大也很有意思,能讓多任務(wù)并行,但是我在其中一個(gè)任務(wù)里調(diào)用系統(tǒng)函數(shù) sleep() 的時(shí)候,阻塞任務(wù)會(huì)阻止協(xié)程切換,其實(shí)從協(xié)程的實(shí)現(xiàn)原理上來說也是這么回事。

那么,我也就想模擬協(xié)程阻塞,但是不產(chǎn)生阻塞看是否可行。PHP本身只提供了生成器為協(xié)程調(diào)用提供了支撐,如果不依賴擴(kuò)展,沒有提供多線程的程序?qū)崿F(xiàn)方式,沒有java那么強(qiáng)大,可以開子線程進(jìn)行實(shí)現(xiàn)。

我印象中java的子線程是獨(dú)立執(zhí)行且不會(huì)相互阻塞的,所以我在想,PHP既然可以實(shí)現(xiàn)類似于多線程這樣的機(jī)制,那么能不能實(shí)現(xiàn)調(diào)用過程中非阻塞呢?

經(jīng)過這樣一個(gè)實(shí)現(xiàn)和思考,一開始是陷入了一個(gè)誤區(qū)的,是由于PHP原生函數(shù) sleep() 阻塞造成的思維誤區(qū),那就是認(rèn)為要想真正實(shí)現(xiàn)非阻塞或者說實(shí)現(xiàn)異步的話,是必須依賴于語言底層的。

后來,我想明白了一個(gè)道理,既然某個(gè)方法或者函數(shù)在執(zhí)行過程中,會(huì)產(chǎn)生阻塞,那么把當(dāng)前這個(gè)方法換成自定義的,做成非阻塞(相對(duì)于整個(gè)協(xié)程調(diào)度來說)不就行了嗎?比如上面的定時(shí)執(zhí)行我自己實(shí)現(xiàn)了一個(gè)。

而另一方面,協(xié)程調(diào)度本身的目的也是為了把任務(wù)執(zhí)行過程切成盡量小片,從而快速切換執(zhí)行,達(dá)到并行的目的。從這方面來看,協(xié)程應(yīng)該也算是一種程序設(shè)計(jì)思想。

以下是一個(gè)程序切成盡量小片執(zhí)行的例子:
// 一個(gè)簡(jiǎn)單的例子

這個(gè)例子是把原本用 range 生成一個(gè)很大的整型數(shù)組的方式切換為分片執(zhí)行,也就是說在遍歷的時(shí)候再去取到指定的值,從代碼上來看,內(nèi)存消耗相對(duì)于之前來說就非常小了。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/25603.html

相關(guān)文章

  • python協(xié)程與golang協(xié)程區(qū)別

    摘要:進(jìn)程線程和協(xié)程進(jìn)程的定義進(jìn)程,是計(jì)算機(jī)中已運(yùn)行程序的實(shí)體。協(xié)程和線程的關(guān)系協(xié)程是在語言層面實(shí)現(xiàn)對(duì)線程的調(diào)度,避免了內(nèi)核級(jí)別的上下文消耗。和都引入了消息調(diào)度系統(tǒng)模型,來避免鎖的影響和進(jìn)程線程開銷大的問題。 進(jìn)程、線程和協(xié)程 進(jìn)程的定義: 進(jìn)程,是計(jì)算機(jī)中已運(yùn)行程序的實(shí)體。程序本身只是指令、數(shù)據(jù)及其組織形式的描述,進(jìn)程才是程序的真正運(yùn)行實(shí)例。 線程的定義: 操作系統(tǒng)能夠進(jìn)行運(yùn)算調(diào)度的最小單...

    csRyan 評(píng)論0 收藏0
  • 【宇潤(rùn)日常瘋測(cè)-007】Swoole 協(xié)程與傳統(tǒng) fpm 同步模式比較

    摘要:初識(shí)協(xié)程執(zhí)行結(jié)果協(xié)程與同步模式比較我們一直在說協(xié)程適合用于密集場(chǎng)景,在同樣的硬件配置環(huán)境下,它會(huì)比傳統(tǒng)的同步模式承載更多的訪問量。假設(shè)一次查詢?yōu)?,在傳統(tǒng)同步模式下,當(dāng)前進(jìn)程在這的時(shí)間里,是不能做其它操作的。同步模式,耗費(fèi)左右的是。 如果說數(shù)組是 PHP 的精髓,數(shù)組玩得不6的,根本不能算是會(huì)用PHP。那協(xié)程對(duì)于 Swoole 也是同理,不理解協(xié)程去用 Swoole,那就是在瞎用。 首先...

    henry14 評(píng)論0 收藏0
  • 談?wù)凱ython協(xié)程技術(shù)演進(jìn)

    摘要:事件循環(huán)是異步編程的底層基石。對(duì)事件集合進(jìn)行輪詢,調(diào)用回調(diào)函數(shù)等一輪事件循環(huán)結(jié)束,循環(huán)往復(fù)。協(xié)程直接利用代碼的執(zhí)行位置來表示狀態(tài),而回調(diào)則是維護(hù)了一堆數(shù)據(jù)結(jié)構(gòu)來處理狀態(tài)。時(shí)代的協(xié)程技術(shù)主要是,另一個(gè)比較小眾。 Coding Crush Python開發(fā)工程師 主要負(fù)責(zé)豈安科技業(yè)務(wù)風(fēng)險(xiǎn)情報(bào)系統(tǒng)redq。 引言 1.1. 存儲(chǔ)器山 存儲(chǔ)器山是 Randal Bryant 在《深入...

    zhiwei 評(píng)論0 收藏0
  • Python 學(xué)習(xí)筆記 關(guān)于協(xié)程

    摘要:協(xié)程定義協(xié)程是指一個(gè)過程,這個(gè)過程與調(diào)用方協(xié)作,產(chǎn)出由調(diào)用方提供的值。當(dāng)?shù)玫娇刂茩?quán)時(shí),會(huì)阻塞,同時(shí)等待終止。終止協(xié)程的方法該方法致使生成器在暫停的表達(dá)式處拋出異常。 協(xié)程 定義:協(xié)程是指一個(gè)過程,這個(gè)過程與調(diào)用方協(xié)作,產(chǎn)出由調(diào)用方提供的值。(協(xié)程中必定含有一條yield語句) 協(xié)程與生成器類似,都是定義體內(nèi)包含yield關(guān)鍵字的函數(shù)。不過,在協(xié)程中,yield通常出現(xiàn)在表達(dá)式的右邊(例...

    VishKozus 評(píng)論0 收藏0
  • Python 中進(jìn)程、線程、協(xié)程、同步、異步、回調(diào)

    摘要:進(jìn)程和線程究竟是什么東西傳統(tǒng)網(wǎng)絡(luò)服務(wù)模型是如何工作的協(xié)程和線程的關(guān)系和區(qū)別有哪些過程在什么時(shí)間發(fā)生在剛剛結(jié)束的上海站,來自七牛云存儲(chǔ)的高級(jí)工程師許智翔帶來了關(guān)于的分享中的進(jìn)程線程協(xié)程同步異步回調(diào)。使用紅黑樹管理就緒隊(duì)列。 進(jìn)程和線程究竟是什么東西?傳統(tǒng)網(wǎng)絡(luò)服務(wù)模型是如何工作的?協(xié)程和線程的關(guān)系和區(qū)別有哪些?IO過程在什么時(shí)間發(fā)生? 在剛剛結(jié)束的 PyCon2014 上海站,來自七牛云...

    Forest10 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

FullStackDeveloper

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<