成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

GCD介紹(三):Dispatch Sources

移動開發 iOS
本文為大家講述怎樣使用GCD的dispatch source功能來監視文件描述符、計時器、聯結的用戶事件以及其他類似的行為。由于dispatch source完全與dispatch queue相集成,所以你可以使用任意的dispatch queue。你可以將一個dispatch source的句柄在主線程中執行、在全局隊列中并發執行、或者在用戶隊列中串行執行(執行時會將程序的其他模塊的運算考慮在內)。

[[77411]]

何為Dispatch Sources

簡單來說,dispatch source是一個監視某些類型事件的對象。當這些事件發生時,它自動將一個block放入一個dispatch queue的執行例程中。

說的貌似有點不清不楚。我們到底討論哪些事件類型?

下面是GCD 10.6.0版本支持的事件:

  1. Mach port send right state changes.
  2. Mach port receive right state changes.
  3. External process state change.
  4. File descriptor ready for read.
  5. File descriptor ready for write.
  6. Filesystem node event.
  7. POSIX signal.
  8. Custom timer.
  9. Custom event.

這是一堆很有用的東西,它支持所有kqueue所支持的事件(kqueue是什么?見http://en.wikipedia.org/wiki/Kqueue)以及mach(mach是什么?見http://en.wikipedia.org/wiki/Mach_(kernel))端口、內建計時器支持(這樣我們就不用使用超時參數來創建自己的計時器)和用戶事件。

用戶事件

這些事件里面多數都可以從名字中看出含義,但是你可能想知道啥叫用戶事件。簡單地說,這種事件是由你調用dispatch_source_merge_data函數來向自己發出的信號。

這個名字對于一個發出事件信號的函數來說,太怪異了。這個名字的來由是GCD會在事件句柄被執行之前自動將多個事件進行聯結。你可以將數據“拼接”至dispatch source中任意次,并且如果dispatch queue在這期間繁忙的話,GCD只會調用該句柄一次(不要覺得這樣會有問題,看完下面的內容你就明白了)。

用戶事件有兩種: DISPATCH_SOURCE_TYPE_DATA_ADD 和 DISPATCH_SOURCE_TYPE_DATA_OR.用戶事件源有個 unsigned long data屬性,我們將一個 unsigned long傳入 dispatch_source_merge_data。當使用 _ADD版本時,事件在聯結時會把這些數字相加。當使用 _OR版本時,事件在聯結時會把這些數字邏輯與運算。當事件句柄執行時,我們可以使用dispatch_source_get_data函數訪問當前值,然后這個值會被重置為0。

讓我假設一種情況。假設一些異步執行的代碼會更新一個進度條。因為主線程只不過是GCD的另一個dispatch queue而已,所以我們可以將GUI更新工作push到主線程中。然而,這些事件可能會有一大堆,我們不想對GUI進行頻繁而累贅的更新,理想的情況是當主線程繁忙時將所有的改變聯結起來。

用dispatch source就完美了,使用DISPATCH_SOURCE_TYPE_DATA_ADD,我們可以將工作拼接起來,然后主線程可以知道從上一次處理完事件到現在一共發生了多少改變,然后將這一整段改變一次更新至進度條。

啥也不說了,上代碼:

  1. dispatch_source_t source = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatch_get_main_queue()); 
  2. dispatch_source_set_event_handler(source, ^{ 
  3.     [progressIndicator incrementBy:dispatch_source_get_data(source)]; 
  4. }); 
  5. dispatch_resume(source); 
  6.  
  7. dispatch_apply([array count], globalQueue, ^(size_t index) { 
  8.     // do some work on data at index 
  9.     dispatch_source_merge_data(source, 1); 
  10. }); 

(對于這段代碼,我很想說點什么,我第一次用dispatch source時,我糾結了很久很久,真讓人蛋疼:Dispatch source啟動時默認狀態是掛起的,我們創建完畢之后得主動恢復,否則事件不會被傳遞,也不會被執行

假設你已經將進度條的min/max值設置好了,那么這段代碼就完美了。數據會被并發處理。當每一段數據完成后,會通知dispatch source并將dispatch source data加1,這樣我們就認為一個單元的工作完成了。事件句柄根據已完成的工作單元來更新進度條。若主線程比較空閑并且這些工作單元進行的比較慢,那么事件句柄會在每個工作單元完成的時候被調用,實時更新。如果主線程忙于其他工作,或者工作單元完成速度很快,那么完成事件會被聯結起來,導致進度條只在主線程變得可用時才被更新,并且一次將積累的改變更新至GUI。

現在你可能會想,聽起來倒是不錯,但是要是我不想讓事件被聯結呢?有時候你可能想讓每一次信號都會引起響應,什么后臺的智能玩意兒統統不要。啊。。其實很簡單的,別把自己繞進去了。如果你想讓每一個信號都得到響應,那使用dispatch_async函數不就行了。實際上,使用的dispatch source而不使用dispatch_async的唯一原因就是利用聯結的優勢。

內建事件

上面就是怎樣使用用戶事件,那么內建事件呢?看看下面這個例子,用GCD讀取標準輸入:

  1.     dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); 
  2.     dispatch_source_t stdinSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_READ, 
  3.                                                            STDIN_FILENO, 
  4.                                                            0, 
  5. globalQueue); 
  6.     dispatch_source_set_event_handler(stdinSource, ^{ 
  7.         char buf[1024]; 
  8.         int len = read(STDIN_FILENO, buf, sizeof(buf)); 
  9.         if(len > 0) 
  10.             NSLog(@"Got data from stdin: %.*s", len, buf); 
  11.     }); 
  12.     dispatch_resume(stdinSource); 

簡單的要死!因為我們使用的是全局隊列,句柄自動在后臺執行,與程序的其他部分并行,這意味著對這種情況的提速:事件進入程序時,程序正在處理其他事務。

這是標準的UNIX方式來處理事務的好處,不用去寫loop。如果使用經典的 read調用,我們還得萬分留神,因為返回的數據可能比請求的少,還得忍受無厘頭的“errors”,比如 EINTR (系統調用中斷)。使用GCD,我們啥都不用管,就從這些蛋疼的情況里解脫了。如果我們在文件描述符中留下了未讀取的數據,GCD會再次調用我們的句柄。

對于標準輸入,這沒什么問題,但是對于其他文件描述符,我們必須考慮在完成讀寫之后怎樣清除描述符。對于dispatch source還處于活躍狀態時,我們決不能關閉描述符。如果另一個文件描述符被創建了(可能是另一個線程創建的)并且新的描述符剛好被分配了相同的數字,那么你的dispatch source可能會在不應該的時候突然進入讀寫狀態。de這個bug可不是什么好玩的事兒。

適當的清除方式是使用 dispatch_source_set_cancel_handler,并傳入一個block來關閉文件描述符。然后我們使用 dispatch_source_cancel來取消dispatch source,使得句柄被調用,然后文件描述符被關閉。

使用其他dispatch source類型也差不多。總的來說,你提供一個source(mach port、文件描述符、進程ID等等)的區分符來作為diapatch source的句柄。mask參數通常不會被使用,但是對于 DISPATCH_SOURCE_TYPE_PROC 來說mask指的是我們想要接受哪一種進程事件。然后我們提供一個句柄,然后恢復這個source(前面我加粗字體所說的,得先恢復),搞定。dispatch source也提供一個特定于source的data,我們使用 dispatch_source_get_data函數來訪問它。例如,文件描述符會給出大致可用的字節數。進程source會給出上次調用之后發生的事件的mask。具體每種source給出的data的含義,看man page吧。

計時器

計時器事件稍有不同。它們不使用handle/mask參數,計時器事件使用另外一個函數 dispatch_source_set_timer 來配置計時器。這個函數使用三個參數來控制計時器觸發:

start參數控制計時器第一次觸發的時刻。參數類型是 dispatch_time_t,這是一個opaque類型,我們不能直接操作它。我們得需要 dispatch_time 和  dispatch_walltime 函數來創建它們。另外,常量  DISPATCH_TIME_NOW 和 DISPATCH_TIME_FOREVER 通常很有用。

interval參數沒什么好解釋的。

leeway參數比較有意思。這個參數告訴系統我們需要計時器觸發的精準程度。所有的計時器都不會保證100%精準,這個參數用來告訴系統你希望系統保證精準的努力程度。如果你希望一個計時器沒五秒觸發一次,并且越準越好,那么你傳遞0為參數。另外,如果是一個周期性任務,比如檢查email,那么你會希望每十分鐘檢查一次,但是不用那么精準。所以你可以傳入60,告訴系統60秒的誤差是可接受的。

這樣有什么意義呢?簡單來說,就是降低資源消耗。如果系統可以讓cpu休息足夠長的時間,并在每次醒來的時候執行一個任務集合,而不是不斷的醒來睡去以執行任務,那么系統會更高效。如果傳入一個比較大的leeway給你的計時器,意味著你允許系統拖延你的計時器來將計時器任務與其他任務聯合起來一起執行。

總結

現在你知道怎樣使用GCD的dispatch source功能來監視文件描述符、計時器、聯結的用戶事件以及其他類似的行為。由于dispatch source完全與dispatch queue相集成,所以你可以使用任意的dispatch queue。你可以將一個dispatch source的句柄在主線程中執行、在全局隊列中并發執行、或者在用戶隊列中串行執行(執行時會將程序的其他模塊的運算考慮在內)。

下一篇我會討論如何對dispatch queue進行掛起、恢復、重定目標操作;如何使用dispatch semaphore;如何使用GCD的一次性初始化功能。

責任編輯:閆佳明 來源: dreamingwish
相關推薦

2013-07-15 15:51:32

iOS多線程GCD基本概念Dispatch Qu

2013-07-15 16:28:15

iOS多線程GCD介紹Dispatch Qu

2013-07-15 16:00:59

2015-06-26 09:29:12

Grand Centr

2013-07-15 15:23:03

iOS多線程GCD

2013-07-15 16:55:54

iOS多線程GCD實戰資源競爭

2023-02-03 14:41:03

地址翻譯模式loongarch

2012-09-20 10:50:34

IBMdw

2011-08-15 11:13:06

IOS開發并發Dispatch Qu

2011-04-26 11:03:36

三星打印機

2010-03-17 17:16:46

Fedora 常用軟件

2011-07-19 10:26:49

Active Dire回收站

2010-04-26 17:15:13

Oracle優化器

2010-04-20 12:00:01

負載均衡技術

2020-12-18 10:04:52

API漏洞應用程序編程接口

2010-04-19 09:31:44

KDE

2010-03-05 11:18:52

Linux shell

2010-03-01 15:21:57

WCF死鎖

2012-01-06 11:13:02

SinatraJava框架

2013-07-19 14:00:13

iOS中BlockiOS開發學習
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩欧美中文在线 | 四季久久免费一区二区三区四区 | 99久久精品免费看国产四区 | 99re视频在线 | 狠狠色狠狠色综合系列 | 国产视频久久久 | 免费观看一级毛片视频 | 91看片在线 | 免费在线视频一区二区 | 亚洲国产精品99久久久久久久久 | 日韩一区二区三区av | www.99精品| 日韩欧美一区二区三区四区 | 九一在线观看 | 国户精品久久久久久久久久久不卡 | 久久亚洲欧美日韩精品专区 | 91精品在线看 | 91精品国产乱码麻豆白嫩 | 99re热这里只有精品视频 | 色综合一区二区三区 | 成人精品一区二区三区中文字幕 | 久草网站| 日本大香伊一区二区三区 | 久久一区二区视频 | 久久久久亚洲精品中文字幕 | 亚洲高清av在线 | 波多野结衣电影一区 | 欧美 日韩 综合 | 黄色操视频 | 亚洲精品中文字幕 | 欧美综合国产精品久久丁香 | 国产精品亚洲精品日韩已方 | 国产69精品久久久久777 | 成人免费视频网站在线观看 | 日本精品一区二区 | 成人在线视频观看 | 久久精品国产一区 | 狠狠躁天天躁夜夜躁婷婷老牛影视 | 色精品 | 久久久久成人精品 | 亚洲欧美激情精品一区二区 |