在JDK1.4.2中加入了對(duì)NIO的支持,掌握其中的Selector個(gè)人認(rèn)為是能實(shí)現(xiàn)好NIO的關(guān)鍵。
Selector是用來(lái)獲取注冊(cè)在其中的channel的相關(guān)事件的發(fā)生,也就是accept,read和write。selector中有3個(gè)key set。
key set:包含代表所有在其中注冊(cè)的channel,可以通過(guò)selector.keys()得到。
selected-key set:包含所有被檢測(cè)到有關(guān)注的操作已經(jīng)就緒的key,通過(guò)selector.selectedKeys得到。
cancelled-key set:包含所有已經(jīng)被cancel,但是還沒(méi)有channel還沒(méi)有deregister的key,這個(gè)集合是不能直接被訪問(wèn)的。
key通過(guò)調(diào)用channel的register方法被加入到key set中。被取消的key在select的時(shí)候會(huì)被從相應(yīng)的key set中移除。
key set自身是不可以直接進(jìn)行修改的。
無(wú)論是通過(guò)調(diào)用channel的close方法還是調(diào)用key的cancel方法,key都會(huì)被放置到canceled-key set中。取消的key會(huì)將其channel在下一次select時(shí)將注冊(cè)撤銷,同時(shí)將key從所有的key set中移除。
key在執(zhí)行select操作時(shí)被加入到selected-key set中。在selected-key
set中的key可以通過(guò)調(diào)用iterator的remove方法,將其從selected-key
set中移除,不能通過(guò)其他的辦法將其從selected-key set中移除。
通過(guò)selector的3個(gè)方法select(阻塞選擇,直到有關(guān)心的事件發(fā)生時(shí)退出阻塞),selectNow(不阻塞選擇),select(long)(指定超時(shí)選擇,超時(shí)到達(dá)或者有關(guān)心事件發(fā)生時(shí)退出阻塞),來(lái)獲取關(guān)心事件的發(fā)生。其執(zhí)行步驟分為以下3步:
1、將存在于canceled-key set中的key從所有的key set中移除,撤銷注冊(cè)的channel,清空canceled-key set。
2、地層操作系統(tǒng)檢查是否有關(guān)心的事件發(fā)生,當(dāng)有關(guān)心的事件發(fā)生時(shí),首先檢查channel的key是否已經(jīng)存在于selected-key
set中,如果不存在,則將其加入到selected-key set中去,同時(shí)修改key的ready-operation
set來(lái)表明當(dāng)前ready的操作,而以前存在于ready-operation
set中的信息會(huì)被刪除。如果對(duì)應(yīng)的key已經(jīng)存在于selected-key set中,這直接修改其ready-operation
set來(lái)表明當(dāng)前ready的操作,刪除原來(lái)ready-operation set中的信息。
3、如果在第二步中有加入到canceled-key set中的key,在這一步會(huì)執(zhí)行第一步的操作。
selector自身是線程安全的,而他的key
set卻不是。在一次選擇發(fā)生的過(guò)程中,對(duì)于key的關(guān)心事件的修改要等到下一次select的時(shí)候才會(huì)生效。
另外,key和其代表的channel有可能在任何時(shí)候被cancel和close。因此存在于key
set中的key并不代表其key是有效的,也不代表其channel是open的。如果key有可能被其他的線程取消或關(guān)閉channel,程序必須小
心的同步檢查這些條件。
阻塞了的select可以通過(guò)調(diào)用selector的wakeup方法來(lái)喚醒。
posted on 2005-04-26 10:06
非飛 閱讀(17394)
評(píng)論(3) 編輯 收藏 所屬分類:
JAVA 相關(guān)技術(shù)