03-yield小應(yīng)用-《流暢的Python》16.9.2出租車(chē)協(xié)程改編

#出租車(chē)隊(duì)運(yùn)營(yíng)仿真
import collections

#優(yōu)先級(jí)列表,用來(lái)維護(hù)事件列表畦贸,用于實(shí)現(xiàn)將所有事件按照發(fā)生先后順序執(zhí)行
class PriorityList:
    def __init__(self):
        self.list=[]

    #獲取事件隊(duì)列中最早發(fā)生的事件
    def get_first(self):
        #將事件列表中所有發(fā)生事件按照發(fā)生事件排序后輸出排序后的第一個(gè)值
        self.list.sort(key=lambda elem:elem.time)
        return self.list[0]

    #向列表中添加事件
    def add_list(self,event):
        self.list.append(event)

    #從事件隊(duì)列中刪除事件
    def del_list(self,event):
        self.list.remove(event)

    #事件列表是否為空
    def is_empty(self):
        return self.list==[]

#基本單位
#time 事件發(fā)生事件  proc 所屬出租車(chē)協(xié)程  action 出租車(chē)活動(dòng)事件
Event=collections.namedtuple('Event','time proc action')

#出租車(chē)協(xié)程楞捂,模擬出租車(chē)的各種活動(dòng)
def taxi_process(ident,trips,start_time=0):
    """
    每次改變狀態(tài)時(shí)創(chuàng)建事件,將控制權(quán)讓給仿真器
    """
    time=yield Event(start_time,ident,'level garage')
    for i in range(trips):
        time=yield Event(time,ident,'pick up passenger')
        time=yield Event(time,ident,'drop off passenger')

    yield Event(time,ident,'going home')

#測(cè)試taxi_process,模擬出租車(chē)運(yùn)行
def main_01():
    taxi=taxi_process(ident=13,trips=2,start_time=0)

    current_state=taxi.send(None) #預(yù)激出租車(chē)協(xié)程
    print(current_state)  #level garge 開(kāi)出車(chē)庫(kù)

    #第1次接客(發(fā)生在離開(kāi)車(chē)庫(kù)7分鐘后),載客行駛事件23分鐘
    current_state=taxi.send(current_state.time+7)
    print(current_state) #pick up passenger 接到乘客

    current_state=taxi.send(current_state.time+23)
    print(current_state) #drop off passenger 卸載乘客

    #第2次接客(發(fā)生在離開(kāi)車(chē)庫(kù)5分鐘后),載客行駛時(shí)間48分鐘
    current_state=taxi.send(current_state.time+5)
    print(current_state) #pick up passenger 接到乘客

    current_state=taxi.send(current_state.time+48)
    print(current_state) #drop off passenger 卸載乘客

    #返回車(chē)庫(kù)(發(fā)生在最后一次卸載乘客10分鐘后)
    current_state=taxi.send(current_state.time+10)
    print(current_state) #going home 返回車(chē)庫(kù)

    try:
        current_state=taxi.send(current_state.time+1)
        print(current_state)
    except StopIteration as e:
        print('此時(shí)出租車(chē)協(xié)程執(zhí)行完畢胶坠,拋出StopInteration')

"""
執(zhí)行:
    main_01()
    
輸出結(jié)果如下:

    Event(time=0, proc=13, action='level garage')
    Event(time=7, proc=13, action='pick up passenger')
    Event(time=30, proc=13, action='drop off passenger')
    Event(time=35, proc=13, action='pick up passenger')
    Event(time=83, proc=13, action='drop off passenger')
    Event(time=93, proc=13, action='going home')
    此時(shí)出租車(chē)協(xié)程執(zhí)行完畢沈善,拋出StopInteration
"""

#出租車(chē)模擬類(lèi)
import random

class Simulator:
    #計(jì)算指定aciton到下一個(gè)action之間的耗時(shí)
    def compute_duration(self,previous_action):
        if previous_action=='drop off passenger':
            return random.choice([1,2])
        elif previous_action=='pick up passenger':
            return random.choice([7,8])
        elif previous_action=='level garage':
            return random.choice([10,15])
        elif previous_action=='going home':
            return random.choice([20,30])
        else:
            raise Exception('Unknown Previoud_Action')

    def __init__(self,procs_dict):
        self.events=PriorityList()
        self.procs=procs_dict

    #仿真系統(tǒng)主邏輯
    def run(self,end_time):
        #預(yù)激每一個(gè)出租車(chē)協(xié)程
        for _,proc in self.procs.items():
            #每個(gè)出租車(chē)協(xié)程的第一個(gè)事件
            first_event=proc.send(None) #預(yù)激
            #將每個(gè)出租車(chē)協(xié)程的第一個(gè)事件放入事件列表
            self.events.add_list(first_event)


        #這個(gè)仿真系統(tǒng)的事件主循環(huán)
        sim_time=0
        #遍歷并處理每個(gè)事件
        while sim_time<end_time:#判斷運(yùn)行時(shí)間是否超時(shí)
            #事件列表為空,所有事件執(zhí)行完畢
            if self.events.is_empty():
                print('******** end all events ********')
                return

            current_event=self.events.get_first() #獲取事件列表中最早發(fā)生的事件

            #處理當(dāng)前事件(此處用打印來(lái)模擬處理過(guò)程)
            sim_time,proc_id,previous_action=current_event
            print('taxi:',proc_id,proc_id*'    ',current_event) #打印當(dāng)前事件
            #從事件標(biāo)記列表中移除已處理過(guò)的事件
            self.events.del_list(current_event)

            active_proc=self.procs[proc_id] #獲取這個(gè)事件所屬的出租車(chē)協(xié)程
            next_time=sim_time+self.compute_duration(previous_action) #計(jì)算下一個(gè)事件的發(fā)生時(shí)間

            try:
                next_event=active_proc.send(next_time) #向出租車(chē)協(xié)程發(fā)送參數(shù)(參數(shù)表示下次事件發(fā)生的時(shí)間)
            except StopIteration:
                #如果觸發(fā)這個(gè)異常表示出租車(chē)協(xié)程執(zhí)行完畢
                del self.procs[proc_id]
            else:
                #將新事件添加到事件列表
                self.events.add_list(next_event)
        else:
            #執(zhí)行時(shí)間超時(shí)后的處理
            print('---->執(zhí)行超時(shí)<----')


#測(cè)試出租車(chē)協(xié)程能否正常執(zhí)行
def main_02():
    data_dict={
        1:taxi_process(ident=1,trips=3,start_time=0),
        2:taxi_process(ident=2,trips=4,start_time=30),
        3:taxi_process(ident=3,trips=2,start_time=13),
    }

    sim1=Simulator(data_dict)
    sim1.run(100) #時(shí)間足夠執(zhí)行完成所有的出租車(chē)協(xié)程

'''
執(zhí)行:
    main_02()
輸出結(jié)果:
    taxi: 1      Event(time=0, proc=1, action='level garage')
    taxi: 1      Event(time=10, proc=1, action='pick up passenger')
    taxi: 3              Event(time=13, proc=3, action='level garage')
    taxi: 1      Event(time=17, proc=1, action='drop off passenger')
    taxi: 1      Event(time=18, proc=1, action='pick up passenger')
    taxi: 1      Event(time=26, proc=1, action='drop off passenger')
    taxi: 3              Event(time=28, proc=3, action='pick up passenger')
    taxi: 1      Event(time=28, proc=1, action='pick up passenger')
    taxi: 2          Event(time=30, proc=2, action='level garage')
    taxi: 1      Event(time=35, proc=1, action='drop off passenger')
    taxi: 3              Event(time=36, proc=3, action='drop off passenger')
    taxi: 1      Event(time=37, proc=1, action='going home')
    taxi: 3              Event(time=38, proc=3, action='pick up passenger')
    taxi: 2          Event(time=40, proc=2, action='pick up passenger')
    taxi: 3              Event(time=46, proc=3, action='drop off passenger')
    taxi: 2          Event(time=47, proc=2, action='drop off passenger')
    taxi: 3              Event(time=48, proc=3, action='going home')
    taxi: 2          Event(time=48, proc=2, action='pick up passenger')
    taxi: 2          Event(time=56, proc=2, action='drop off passenger')
    taxi: 2          Event(time=57, proc=2, action='pick up passenger')
    taxi: 2          Event(time=64, proc=2, action='drop off passenger')
    taxi: 2          Event(time=66, proc=2, action='pick up passenger')
    taxi: 2          Event(time=74, proc=2, action='drop off passenger')
    taxi: 2          Event(time=75, proc=2, action='going home')
    ******** end all events ********
'''

#測(cè)試設(shè)置超時(shí)時(shí)間是否有效
def main_03():
    data_dict={
        1:taxi_process(ident=1,trips=3,start_time=0),
        2:taxi_process(ident=2,trips=4,start_time=30),
        3:taxi_process(ident=3,trips=2,start_time=13),
    }

    sim1=Simulator(data_dict)
    sim1.run(30) #時(shí)間不足以執(zhí)行完成所有的出租車(chē)協(xié)程
'''
執(zhí)行
    main_03()
輸出結(jié)果:
    taxi: 1      Event(time=0, proc=1, action='level garage')
    taxi: 3              Event(time=13, proc=3, action='level garage')
    taxi: 1      Event(time=15, proc=1, action='pick up passenger')
    taxi: 1      Event(time=22, proc=1, action='drop off passenger')
    taxi: 3              Event(time=23, proc=3, action='pick up passenger')
    taxi: 1      Event(time=23, proc=1, action='pick up passenger')
    taxi: 2          Event(time=30, proc=2, action='level garage')
    taxi: 3              Event(time=30, proc=3, action='drop off passenger')
    taxi: 1      Event(time=30, proc=1, action='drop off passenger')
    ---->執(zhí)行超時(shí)<---- 
'''

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末绳矩,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子埋酬,更是在濱河造成了極大的恐慌,老刑警劉巖拳球,帶你破解...
    沈念sama閱讀 212,029評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件珍特,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)酬姆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,395評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)奥溺,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人浮定,你說(shuō)我怎么就攤上這事¤胱洌” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,570評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵建蹄,是天一觀的道長(zhǎng)裕偿。 經(jīng)常有香客問(wèn)我,道長(zhǎng)击费,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,535評(píng)論 1 284
  • 正文 為了忘掉前任谆棱,我火速辦了婚禮,結(jié)果婚禮上垃瞧,老公的妹妹穿的比我還像新娘。我一直安慰自己个从,他們只是感情好歪沃,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,650評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著沪曙,像睡著了一般。 火紅的嫁衣襯著肌膚如雪碳默。 梳的紋絲不亂的頭發(fā)上贾陷,一...
    開(kāi)封第一講書(shū)人閱讀 49,850評(píng)論 1 290
  • 那天髓废,我揣著相機(jī)與錄音,去河邊找鬼慌洪。 笑死柔逼,一個(gè)胖子當(dāng)著我的面吹牛割岛,可吹牛的內(nèi)容都是我干的愉适。 我是一名探鬼主播癣漆,決...
    沈念sama閱讀 39,006評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼癌蓖!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起租副,我...
    開(kāi)封第一講書(shū)人閱讀 37,747評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤较性,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后赞咙,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,207評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡攀操,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,536評(píng)論 2 327
  • 正文 我和宋清朗相戀三年速和,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片颠放。...
    茶點(diǎn)故事閱讀 38,683評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖慈迈,靈堂內(nèi)的尸體忽然破棺而出省有,到底是詐尸還是另有隱情谴麦,我是刑警寧澤,帶...
    沈念sama閱讀 34,342評(píng)論 4 330
  • 正文 年R本政府宣布舷蟀,位于F島的核電站,受9級(jí)特大地震影響野宜,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜匈子,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,964評(píng)論 3 315
  • 文/蒙蒙 一闯袒、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧政敢,春花似錦、人聲如沸喷户。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,772評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至恼五,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間灾馒,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,004評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工轨功, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人古涧。 一個(gè)月前我還...
    沈念sama閱讀 46,401評(píng)論 2 360
  • 正文 我出身青樓花盐,卻偏偏與公主長(zhǎng)得像菇爪,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子凳宙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,566評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容