在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

6種I/O模式告訴你協(xié)程的作用

Q4MP_gh_c472c21 ? 來源:碼農(nóng)的荒島求生 ? 作者:陸小風(fēng) ? 2022-05-24 15:23 ? 次閱讀

大家好,我是小風(fēng)哥,今天我們來聊聊協(xié)程的作用。

假設(shè)磁盤上有10個文件,你需要讀取的內(nèi)存,那么你該怎么用代碼實現(xiàn)呢?

在接著往下看之前,先自己想一想這個問題,看看自己能想出幾種方法,各自有什么樣的優(yōu)缺點。想清楚了嗎(還在看嗎),想清楚了我們繼續(xù)往下看。

最簡單的方法——串行這可能是大多數(shù)同學(xué)都能想到的最簡單方法,那就是一個一個的讀取,讀完一個接著讀下一個。用代碼表示是這樣的:
for file in files:  result = file.read()  process(result)
是不是非常簡單,我們假設(shè)每個文件讀取需要1分鐘,那么10個文件總共需要10分鐘才能讀取完成。這種方法有什么問題呢?實際上,這種方法只有一個問題,那就是除此之外,其它都是優(yōu)點,比如:
  1. 代碼簡單,容易理解

  2. 可維護性好,這代碼交給誰都能維護的了(論程序員的核心競爭力在哪里)

那么,慢的問題又該怎么解決呢?有的同學(xué)可能已經(jīng)想到了,為啥要一個一個讀取呢?并行讀取不就可以加快速度了嗎。
稍好的方法,并行那么,該怎么并行讀取文件呢?顯然,地球人都知道,線程就是用來并行的。我們可以同時開啟10個線程,每個線程中讀取一個文件。用代碼實現(xiàn)就是這樣的:

		def read_and_process(file): result = file.read() process(result) def main(): files = [fileA,fileB,fileC......] for file in files: create_thread(read_and_process, file).run() # 等待這些線程執(zhí)行完成怎么樣,是不是也非常簡單。那么這種方法有什么問題嗎?在開啟10個線程這種問題規(guī)模下沒有問題。現(xiàn)在我們把問題難度加大,假設(shè)有10000個文件,需要處理該怎么辦呢?有的同學(xué)可能想10個文件和10000個文件有什么區(qū)別嗎,直接創(chuàng)建10000個線程去讀不可以嗎?實際上,這里的問題其實是說創(chuàng)建多個線程有沒有什么問題。我們知道,雖然線程號稱“輕量級進程”,雖然是輕量級但當數(shù)量足夠可觀時依然會有性能問題。這里的問題主要有這樣幾個方面:
  1. 創(chuàng)建線程需要消耗系統(tǒng)資源,像內(nèi)存等(想一想為什么?)

  2. 調(diào)度開銷,尤其是當線程數(shù)量較多且都比較繁忙時(同樣想一想為什么?)

  3. 創(chuàng)建多個線程不一定能加快I/O(如果此時設(shè)備處理能力已經(jīng)飽和)

既然線程有這樣那樣的問題,那么還有沒有更好的方法?答案是肯定的,并行編程不一定只能依賴線程這種技術(shù),關(guān)于并發(fā)編程可以用哪些技術(shù)實現(xiàn)的詳細討論請參考《高性能服務(wù)器是如何實現(xiàn)的》。這里的答案就是基于事件驅(qū)動編程技術(shù)。
事件驅(qū)動 + 異步沒錯,即使在單個線程中,使用事件驅(qū)動+異步也可以實現(xiàn)IO并行處理,Node.js就是非常典型的例子。為什么單線程也可以做到并行呢?這是基于這樣兩個事實:
  1. 相對于CPU的處理速度來說,IO是非常慢的

  2. IO不怎么需要計算資源

因此,當我們發(fā)起IO操作后為什么要一直等著IO執(zhí)行完成呢?在IO執(zhí)行完之前的這段時間處理其它IO難道不香嗎?這就是為什么單線程也可以并行處理多個IO的本質(zhì)所在。回到我們的例子,該怎樣用事件驅(qū)動+異步來改造上述程序呢?實際上非常簡單。首先,我們需要創(chuàng)建一個event loop,這個非常簡單:

		event_loop = EventLoop()然后,我們需要往event loop中加入原材料,也就是需要監(jiān)控的event,就像這樣:

		def add_to_event_loop(event_loop, file): file.asyn_read() # 文件異步讀取 event_loop.add(file)注意,當執(zhí)行file.asyn_read這行代碼時會立即返回,不會阻塞線程,當這行代碼返回時可能文件還沒有真正開始讀取,這就是所謂的異步。file.asyn_read這行代碼的真正目的僅僅是發(fā)起IO,而不是等待IO執(zhí)行完成。此后,我們將該IO放到event loop中進行監(jiān)控,也就是event_loop.add(file)這行代碼的作用。一切準備就緒,接下來就可以等待event的到來了:

		while event_loop: file = event_loop.wait_one_IO_ready() process(file.result)我們可以看到,event_loop會一直等待直到有文件讀取完成(event_loop.wait_one_IO_ready())。這時,我們就能得到讀完的文件了,接下來處理即可。全部代碼如下所示:
 def add_to_event_loop(event_loop, file):   file.asyn_read() # 文件異步讀取   event_loop.add(file)
def main():files=[fileA,fileB,fileC...]  event_loop = EventLoop()  for file in files:      add_to_event_loop(event_loop, file)        while event_loop:     file = event_loop.wait_one_IO_ready()     process(file.result)

多線程 VS 單線程 + event loop接下來,我們看下程序執(zhí)行的效果。在多線程情況下,假設(shè)有10個文件,每個文件讀取需要1秒,那么很簡單,并行讀取10個文件需要1秒。那么,對于單線程+event loop呢?我們再次看下event loop + 異步版本的代碼:

		def add_to_event_loop(event_loop, file): file.asyn_read() # 文件異步讀取 event_loop.add(file) def main(): files = [fileA,fileB,fileC......] event_loop = EventLoop() for file in files: add_to_event_loop(event_loop, file)  while event_loop: file = event_loop.wait_one_IO_ready() process(file.result)對于add_to_event_loop,由于文件異步讀取,因此該函數(shù)可以瞬間執(zhí)行完成,真正耗時的函數(shù)其實就是event loop的等待函數(shù),也就是這樣:

		file = event_loop.wait_one_IO_ready()我們知道,一個文件的讀取耗時是1秒,因此該函數(shù)在1s后才能返回,但是,但是,接下來是重點。但是,雖然該函數(shù)wait_one_IO_ready會等待1s,不要忘了,我們利用這兩行代碼同時發(fā)起了10個IO操作請求。
for file in files:  add_to_event_loop(event_loop, file)
因此,在event_loop.wait_one_IO_ready等待的1s期間,剩下的9個IO也完成了。也就是說,event_loop.wait_one_IO_ready函數(shù)只是在第一次循環(huán)時會等待1s,但此后的9次循環(huán)會直接返回,原因就在于剩下的9個IO也完成了。因此,整個程序的執(zhí)行耗時也是1秒。是不是很神奇,我們只用一個線程就達到了10個線程的效果。這就是event loop + 異步的威力所在。
一個好聽的名字:Reactors模式本質(zhì)上,我們上述給出的event loop簡單代碼片段做的事情本質(zhì)上和生物一樣:給出刺激,做出反應(yīng)。我們這里的給出event,然后處理event。這本質(zhì)上就是所謂的Reactors模式。現(xiàn)在你應(yīng)該明白所謂的Reactors模式是怎么一回事了吧。所謂的一些看上去復(fù)雜的異步框架,其核心不過就是這里給出的代碼片段,只是這些框架可以支持更加復(fù)雜的多階段任務(wù)處理,以及各種類型的IO。而我們這里給出的代碼片段,只能處理文件讀取這一類IO。
把回調(diào)也加進來如果我們需要處理各種類型的IO上述代碼片段會有什么問題嗎?問題就在于上述代碼片段就不會這么簡單了,針對不同類型會有不同的處理方法。因此,上述process方法需要判斷IO類型然后有針對性的處理,這會使得代碼越來越復(fù)雜,越來越難以維護。幸好我們也有應(yīng)對策略,這就是回調(diào)。關(guān)于回調(diào)函數(shù),請參考這篇《程序員應(yīng)如何理解回調(diào)函數(shù)》。我們可以把IO完成后的處理任務(wù)封裝到回調(diào)函數(shù)中,然后和IO一并注冊到event loop。就像這樣:

		def IO_type_1(event_loop, io): io.start()  def callback(result): process_IO_type_1(result)  event_loop.add((io, callback))這樣,event_loop在檢測到有IO完成后就可以把該IO和關(guān)聯(lián)的callback處理函數(shù)一并檢索出來,直接調(diào)用callback函數(shù)就可以了。

		while event_loop: io, callback = event_loop.wait_one_IO_ready() callback(io.result)看到了吧,這樣event_loop內(nèi)部就極其簡潔了,even_loop根本就不關(guān)心該怎么處理該IO結(jié)果,這是注冊的callback該關(guān)心的事情,event_loop需要做的僅僅就是拿到event以及相應(yīng)的處理函數(shù)callback,然后調(diào)用該callback函數(shù)就可以了。現(xiàn)在我們可以同單線程來并發(fā)編程了,也使用callback對IO處理進行了抽象,使得代碼更加容易維護,想想看還有沒有什么問題?
		
回調(diào)函數(shù)的問題雖然回調(diào)函數(shù)使得event loop內(nèi)部更加簡潔,但依然有其它問題,讓我們來仔細看看回調(diào)函數(shù):

		def start_IO_type_1(event_loop, io): io.start()  def callback(result): process_IO_type_1(result)  event_loop.add((io, callback))從上述代碼中你能看到什么問題嗎?在上述代碼中,一次IO處理過程被分為了兩個部分:
  1. 發(fā)起IO

  2. IO處理

其中,第2部分放到了回調(diào)函數(shù)中,這樣的異步處理天然不容易理解,這和我們熟悉的發(fā)起IO,等待IO完成、處理IO結(jié)果的同步模塊有很大差別。這里的給的例子很簡單,所以你可能不以為意,但是當處理的任務(wù)非常復(fù)雜時,可能會出現(xiàn)回調(diào)函數(shù)中嵌套回調(diào)函數(shù),也就是回調(diào)地獄,這樣的代碼維護起來會讓你懷疑為什么要稱為一名苦逼的碼農(nóng)。
問題出在哪里讓我們再來仔細的看看問題出在了哪里?同步編程模式下很簡單,但是同步模式下發(fā)起IO,線程會被阻塞,這樣我們就不得不創(chuàng)建多個線程,但是創(chuàng)建過多線程又會有性能問題。這樣為了發(fā)起IO后不阻塞當前線程我們就不得不采用異步編程+event loop。在這種模式下,異步發(fā)起IO不會阻塞調(diào)用線程,我們可以使用單線程加異步編程的方法來實現(xiàn)多線程效果,但是在這種模式下處理一個IO的流程又不得不被拆分成兩部分,這樣的代碼違反程序員直覺,因此難以維護。那么很自然的,有沒有一種方法既能有同步編程的簡單理解又會有異步編程的非阻塞呢?答案是肯定的,這就是協(xié)程。關(guān)于協(xié)程請參考《程序員應(yīng)如何理解協(xié)程》。
Finally!終于到了協(xié)程利用協(xié)程,我可以以同步的形式來異步編程。這是什么意思呢?我們之所以采用異步編程是為了發(fā)起IO后不阻塞當前線程,而是用協(xié)程,程序員可以自行決定在什么時刻掛起當前協(xié)程,這樣也不會阻塞當前線程。而協(xié)程最棒的一點就在于掛起后可以暫存執(zhí)行狀態(tài)恢復(fù)運行后可以在掛起點繼續(xù)運行,這樣我們就不再需要像回調(diào)那樣將一個IO的處理流程拆分成兩部分了。因此,我們可以在發(fā)起異步IO,這樣不會阻塞當前線程,同時在發(fā)起異步IO后掛起當前協(xié)程,當IO完成后恢復(fù)該協(xié)程的運行。這樣一來,我們就可以實現(xiàn)同步的方式來異步編程了。接下來,我們就用協(xié)程來改造一下回調(diào)版本的IO處理方式:

		def start_IO_type_1(io): io.start() # IO異步請求 yield # 暫停當前協(xié)程  process_IO_type_1(result) # 處理返回結(jié)果此后,我們要把該協(xié)程放到event loop中監(jiān)控起來:

		def add_to_event_loop(io, event_loop): coroutine = start_IO_type_1(io) next(coroutine) event_loop.add(coroutine)最后,當IO完成后event loop檢索出相應(yīng)的協(xié)程并恢復(fù)其運行:

		while event_loop: coroutine = event_loop.wait_one_IO_ready() next(coroutine)現(xiàn)在你應(yīng)該看出來了吧,上述代碼中沒有回調(diào),也沒有把處理IO的流程拆成兩部分,整體的代碼都是以同步的方式來編寫,最棒的是依然能達到異步的效果。實際上你會看到,采用協(xié)程后我們依然需要基于事件編程的event loop,因為本質(zhì)上協(xié)程并沒有改變IO的異步處理本質(zhì),只要IO是異步處理的那么我們就必須依賴event loop來監(jiān)控IO何時完成,只不過我們采用協(xié)程消除了對回調(diào)的依賴,整體編程方式上還是采用程序員最熟悉也最容易理解的同步方式。
		
總結(jié)看上去簡簡單單的IO,實際上一點都不簡單。為了高效進行IO操作,我們采用的技術(shù)是這樣演進的:
  1. 單線程串行 + 阻塞式IO(同步)

  2. 多線程并行 + 阻塞式IO(并行)

  3. 單線程 + 非阻塞式IO(異步) + event loop

  4. 單線程 + 非阻塞式IO(異步) + event loop + 回調(diào)

  5. Reactor模式(更好的單線程 + 非阻塞式IO+ event loop + 回調(diào))

  6. 單線程 + 非阻塞式IO(異步) + event loop + 協(xié)程

最終,我們采用協(xié)程技術(shù)獲取到了異步編程的高效以及同步編程的簡單理解,這也是當今高性能服務(wù)器常用的一種技術(shù)組合。希望這篇文章能對你理解高效IO有所幫助。

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4891

    瀏覽量

    70342
  • 模式
    +關(guān)注

    關(guān)注

    0

    文章

    65

    瀏覽量

    13599

原文標題:6種I/O模式告訴你,協(xié)程到底有什么用?

文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦
    熱點推薦

    能否告訴我在CapSense按鈕觸發(fā)時是否有可能重新配置I/O的操作?

    能否告訴我在 CapSense 按鈕觸發(fā)時是否有可能重新配置 I/O 的操作? 我想使用一個 CapSense 按鈕并啟用/禁用電路上的另一個 IC。 這意味著,如果按下 CapSense 按鈕,輸出將永久保持高電平,而如果再次
    發(fā)表于 05-29 07:58

    MAX7325 I2C端口擴展器,提供8路推挽式I/O和8個漏極開路I/O技術(shù)手冊

    MAX7325 2線串行接口外設(shè)具有16路I/O端口。其中8路為推挽輸出,另外8路為I/O端口,帶有可選擇的內(nèi)部上拉和瞬態(tài)檢測功能。8路I/
    的頭像 發(fā)表于 05-22 15:27 ?165次閱讀
    MAX7325 <b class='flag-5'>I</b>2C端口擴展器,提供8路推挽式<b class='flag-5'>I</b>/<b class='flag-5'>O</b>和8個漏極開路<b class='flag-5'>I</b>/<b class='flag-5'>O</b>技術(shù)手冊

    LuatOS協(xié)深度解析:小白也能10分鐘學(xué)會,代碼效率直接起飛!

    嵌入式開發(fā)如何兼顧效率與簡潔?LuatOS協(xié)給出完美答案!它用類線程的語法封裝異步邏輯,讓多任務(wù)開發(fā)像單線程一樣簡單。本文用圖文并茂的方式拆解協(xié)原理,10分鐘帶你輕松入門! ? L
    的頭像 發(fā)表于 04-10 15:23 ?155次閱讀
    LuatOS<b class='flag-5'>協(xié)</b><b class='flag-5'>程</b>深度解析:小白也能10分鐘學(xué)會,代碼效率直接起飛!

    10分鐘上手寫代碼,LuatOS協(xié)輕松掌握!

    10分鐘學(xué)會LuatOS協(xié),從此的程序也能像通勤族利用碎片時間一樣游刃有余?,F(xiàn)在就去動手試一試,開啟異步編程新體驗! 寫給第一次聽說協(xié)
    的頭像 發(fā)表于 04-10 15:18 ?220次閱讀
    10分鐘上手寫代碼,LuatOS<b class='flag-5'>協(xié)</b><b class='flag-5'>程</b>輕松掌握!

    進程、線程、協(xié)傻傻分不清?一文帶你徹底扒光它們的\"底褲\"!

    電動車必須充電才能跑) 職場類比:** 是個超級斜杠青年(主線程),同時干著: 切水果(協(xié)A) 燒水(協(xié)B) 回復(fù)微信(
    發(fā)表于 03-26 09:27

    I/O接口與I/O端口的區(qū)別

    在計算機系統(tǒng)中,I/O接口與I/O端口是實現(xiàn)CPU與外部設(shè)備數(shù)據(jù)交換的關(guān)鍵組件,它們在功能、結(jié)構(gòu)、作用及運作機制上均存在顯著差異,卻又相互協(xié)
    的頭像 發(fā)表于 02-02 16:00 ?1239次閱讀

    數(shù)據(jù)I/O模塊的概念、特點以及作用

    ? 本文簡單介紹了數(shù)據(jù)I/O模塊的概念、特點以及作用。 一、數(shù)據(jù) I/O 模塊是什么 1. 承接內(nèi)外數(shù)據(jù)交互的“橋梁” 數(shù)據(jù)
    的頭像 發(fā)表于 01-21 11:10 ?814次閱讀

    直接I/O

    電子發(fā)燒友網(wǎng)站提供《直接I/O庫.pdf》資料免費下載
    發(fā)表于 10-14 10:55 ?0次下載
    直接<b class='flag-5'>I</b>/<b class='flag-5'>O</b>庫

    物聯(lián)網(wǎng)中常見的I/O擴展電路設(shè)計方案_IIC I/O擴展芯片

    物聯(lián)網(wǎng)系統(tǒng)中為什么要使用 IIC I/O擴展芯片 ??在物聯(lián)網(wǎng)系統(tǒng)中使用IIC(也稱為I2C)I/O擴展芯片的原因主要可以歸結(jié)為以下幾點:
    的頭像 發(fā)表于 09-24 11:29 ?1235次閱讀
    物聯(lián)網(wǎng)中常見的<b class='flag-5'>I</b>/<b class='flag-5'>O</b>擴展電路設(shè)計方案_IIC <b class='flag-5'>I</b>/<b class='flag-5'>O</b>擴展芯片

    保護I/O模塊免受浪涌事件的影響

    電子發(fā)燒友網(wǎng)站提供《保護I/O模塊免受浪涌事件的影響.pdf》資料免費下載
    發(fā)表于 09-21 10:14 ?0次下載
    保護<b class='flag-5'>I</b>/<b class='flag-5'>O</b>模塊免受浪涌事件的影響

    遠程I/O模塊在不同領(lǐng)域的應(yīng)用

    模塊發(fā)揮著至關(guān)重要的作用。它們能夠?qū)崟r收集生產(chǎn)線上各種設(shè)備的運行狀態(tài)和生產(chǎn)數(shù)據(jù),并將這些信息傳輸至中控系統(tǒng)。這使得生產(chǎn)管理人員能夠?qū)崟r掌握生產(chǎn)線的運行情況,及時調(diào)整生產(chǎn)策略,提高生產(chǎn)效率和產(chǎn)品質(zhì)量。同時,遠程I/O模塊還降低
    的頭像 發(fā)表于 09-20 16:43 ?1165次閱讀

    I/O模塊的主要作用有哪些

    遠程I/O模塊是一使能遠程數(shù)據(jù)采集和控制的設(shè)備。通過使用網(wǎng)絡(luò)技術(shù),如現(xiàn)場總線、以太網(wǎng)等,遠程I/O模塊能夠?qū)⑤斎牒洼敵鲂盘杺鬟f給控制系統(tǒng)。
    的頭像 發(fā)表于 09-20 16:41 ?1099次閱讀

    淺談如何克服FPGA I/O引腳分配挑戰(zhàn)

    ,” 以及“ Place I/O Ports Sequentially.”。每種模式提供了將I/O端口分配到引腳的不同分配方式。 利用這些
    發(fā)表于 07-22 00:40

    PLC的I/O點數(shù)是什么意思

    在工業(yè)自動化領(lǐng)域中,可編程邏輯控制器(PLC)扮演著至關(guān)重要的角色。PLC以其高可靠性、易編程性和強大的控制功能,廣泛應(yīng)用于各種自動化系統(tǒng)中。而在PLC的性能參數(shù)中,I/O點數(shù)是一個不可忽視的重要指標。本文將對PLC的I/
    的頭像 發(fā)表于 06-27 11:15 ?7572次閱讀

    PLC的I/O模塊的作用及其重要性

    在工業(yè)自動化領(lǐng)域中,可編程邏輯控制器(PLC)扮演著至關(guān)重要的角色。作為PLC的核心組成部分,I/O(輸入/輸出)模塊不僅連接著PLC與外部設(shè)備,更是實現(xiàn)信息交換的關(guān)鍵橋梁。本文將詳細探討PLC的I/
    的頭像 發(fā)表于 06-19 10:43 ?4708次閱讀
    主站蜘蛛池模板: 福利片第一页 | 国产成人夜间影院在线观看 | 四虎影业| 1024国产高清精品推荐 | 狠狠色狠狠色综合日日32 | 午夜剧场官网 | 日本色www | 中国成人在线视频 | 青青导航| 特黄日韩免费一区二区三区 | 亚洲福利视频网址 | 手机看片中文字幕 | 四虎成人精品在永久在线观看 | 丁香亚洲综合五月天婷婷 | 九九热精品视频在线播放 | 最近2018年中文字幕在线 | 久草亚洲视频 | 狠狠色影院 | 奇米影视亚洲四色8888 | 美女教师一级毛片 | 欧美日韩视频综合一区无弹窗 | 国产69精品久久 | 日韩毛片高清在线看 | 东北美女野外bbwbbw免费 | 天天干妹子| 在线你懂的 | 黄色伊人| 免费啪视频观在线视频在线 | 特黄特级高清免费视频毛片 | 男人不识本网站上遍色站也枉然 | 亚洲一区 在线播放 | 黄页网站在线 | 97影院理论片在线观看 | 69久久夜色精品国产69小说 | 欧美+日本+国产+在线观看 | 日本黄色录象 | 长腿丝袜美女被啪啪 | 免费在线公开视频 | 精品国产三级a∨在线 | 性性性性bbbbxxxx| 黄鳝钻进下面好爽小说 |