netty手動(dòng)關(guān)閉資源的探究

最近突然得知自己的基礎(chǔ)差霞怀,與精英的標(biāo)準(zhǔn)相距甚遠(yuǎn)褥实。的確呀狼,平時(shí)的工作中,有太多機(jī)會允許我運(yùn)用“基礎(chǔ)”以外的東西去完成损离。讓我對“基礎(chǔ)”沒有很重視哥艇,所以決定了馬上亡羊補(bǔ)牢。就先從自己最看重的異步編程開刀吧僻澎,翻閱起了《Netty權(quán)威指南》貌踏。大部分內(nèi)容與看之前的道聽途說、自行猜想沒差多少窟勃。突如其來的如下代碼祖乳,讓我產(chǎn)生了警覺:


1.png

代碼中,由于手動(dòng)打開了本地文件秉氧,所以在flush操作后眷昆,手動(dòng)關(guān)閉了文件。但netty作為一個(gè)異步框架汁咏,總覺得不太可能需要寫出這種同步樣式的代碼亚斋。
我的疑問有兩點(diǎn):1、flush方法返回時(shí)攘滩,可以確保數(shù)據(jù)write完畢帅刊?2、netty沒有提供用于編寫收尾邏輯的回調(diào)接口漂问?
弄清疑問一:
嘗試在互聯(lián)網(wǎng)上搜索赖瞒,發(fā)現(xiàn)并沒有關(guān)于這點(diǎn)的描述,所以干脆自己下載了源碼自己看蚤假。
ChannelHandlerContext接口栏饮,對flush的描述:

2.png

AbstractChannelHandlerContext抽象類中的具體實(shí)現(xiàn):

3.png

意思應(yīng)該是將pipeline中的ChannelOutboundHandler依次調(diào)一遍,參照findContextOutbound的實(shí)現(xiàn):

4.png

可以得知磷仰,ChannelOutboundHandler是從后向前依次調(diào)用的袍嬉。最后一個(gè)調(diào)用的為默認(rèn)的HeadContext:

5.png

HeadContext的flush,調(diào)用的是DefaultChannelPipeline.channel().unsafe().flush():

6.png

7.png

以書中的示例代碼使用的NioServerSocketChannel為例芒划,當(dāng)接收到客戶端請求時(shí)冬竟,創(chuàng)建了NioSocketChannel:

8.png

8+.png

NioSocketChannel中unsafe().flush()的調(diào)用鏈為:
unsafe()[繼承自:AbstractNioByteChannel欧穴、AbstractNioChannel] >>> AbstractChannel.unsafe()
AbstractChannel.unsafe()返回的結(jié)果來自:
NioSocketChannel.newUnsafe()

9.png

NioSocketChannelUnsafe的flush方法[繼承自:NioByteUnsafe、AbstractNioUnsafe泵殴、AbstractUnsafe]:

10.png

調(diào)用的flush0方法:
flush0()[繼承自:NioByteUnsafe涮帘、AbstractNioUnsafe] >>> AbstractUnsafe.flush0()

11.png

調(diào)用的NioSocketChannel.doWrite方法:

12.png

針對示例中的FileRegion,調(diào)用了父類doWrite方法:

13.png

14.png

其中確實(shí)存在針對write未完畢的處理笑诅,調(diào)用用incompleteWrite(true) >>> setOpWrite():

15.png

將SelectionKey.OP_WRITE注冊到selector中调缨,并在寫操作完成后,由selector再觸發(fā)一次flush(NioEventLoop代碼):

16.png

總上吆你,flush方法返回時(shí)弦叶,不保證數(shù)據(jù)write完畢。
弄清疑問二:
DefaultFileRegion的API說明:
Default FileRegion implementation which transfer data from a FileChannel or File. Be aware that the FileChannel will be automatically closed once AbstractReferenceCounted.refCnt() returns 0.
DefaultFileRegion類實(shí)現(xiàn)了FileRegion接口妇多,繼承了AbstractReferenceCounted類伤哺。創(chuàng)建DefaultFileRegion實(shí)例時(shí)可以傳入一個(gè)FileChannel實(shí)現(xiàn)或一個(gè)File實(shí)例,而且在這里使用FileChannel不需要手動(dòng)close者祖。
上文的AbstractNioByteChannel.doWrite方法中立莉,完畢時(shí)調(diào)用了ChannelOutboundBuffer.remove():

17.png

這正是用來釋放ReferenceCounted對象的邏輯。
DefaultFileRegion中也存在關(guān)閉FileChannel的邏輯:

18.png

但RandomAccessFile方法中七问,不只有關(guān)閉FileChannel的邏輯:

19.png

因此RandomAccessFile.close()方法還是需要手動(dòng)調(diào)用的蜓耻。
總終方案:
編寫DefaultFileRegion子類,重寫deallocate方法械巡,額外關(guān)閉RandomAccessFile對象刹淌。(代碼略)

探究感想:
為了論證自己的猜測,確實(shí)花費(fèi)了不少精力讥耗。最大的困難在于本次的論點(diǎn)有勾,沒有被廣大程序員關(guān)注,而且錯(cuò)誤的代碼遍布各種書籍葛账、網(wǎng)站柠衅,讓人直覺上認(rèn)為那些就是對的皮仁。這正讓我聯(lián)想到之前美團(tuán)的招聘細(xì)則中不要“信中醫(yī)的”籍琳。中醫(yī)這里代表的是廣泛留傳下來的傳統(tǒng)中醫(yī)理論知識,不單只運(yùn)用了現(xiàn)代科學(xué)醫(yī)藥手段驗(yàn)證過的中醫(yī)贷祈。中醫(yī)與軟件理論趋急,都十分龐大、且充滿糟粕势誊。對待中醫(yī)和軟件知識呜达,都應(yīng)該持有一種警覺態(tài)度,那些被記載粟耻、流傳下來的手段查近、方案眉踱,真的是正確的嗎?適用在你的場景中依然有效霜威?的確谈喳,實(shí)際運(yùn)用中會有眾多因素,制約著你無法透徹的去驗(yàn)證戈泼。但你的知識獲取方法婿禽,只能是聽信他人嗎。1753年英國海軍軍醫(yī)倫達(dá)大猛,經(jīng)試驗(yàn)得出結(jié)論:檸檬汁可用來治療和預(yù)防壞血病扭倾。這種的結(jié)論,對于航海而言就足夠了挽绩,不需要知道其根本是維生素C的作用膛壹。中醫(yī)知識通過大量隨機(jī)雙盲對照試驗(yàn)來驗(yàn)證。軟件知識唉堪,歸根到底就是代碼恢筝,真有什么不明白的,看一看巨坊、運(yùn)行一下撬槽。你能了解到什么程度,首先由你想了解到什么程度決定趾撵。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末侄柔,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子占调,更是在濱河造成了極大的恐慌暂题,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,627評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件究珊,死亡現(xiàn)場離奇詭異薪者,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)剿涮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評論 3 399
  • 文/潘曉璐 我一進(jìn)店門言津,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人取试,你說我怎么就攤上這事悬槽。” “怎么了瞬浓?”我有些...
    開封第一講書人閱讀 169,346評論 0 362
  • 文/不壞的土叔 我叫張陵初婆,是天一觀的道長。 經(jīng)常有香客問我,道長磅叛,這世上最難降的妖魔是什么屑咳? 我笑而不...
    開封第一講書人閱讀 60,097評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮弊琴,結(jié)果婚禮上乔宿,老公的妹妹穿的比我還像新娘。我一直安慰自己访雪,他們只是感情好详瑞,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,100評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著臣缀,像睡著了一般坝橡。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上精置,一...
    開封第一講書人閱讀 52,696評論 1 312
  • 那天计寇,我揣著相機(jī)與錄音,去河邊找鬼脂倦。 笑死番宁,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的赖阻。 我是一名探鬼主播蝶押,決...
    沈念sama閱讀 41,165評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼火欧!你這毒婦竟也來了棋电?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 40,108評論 0 277
  • 序言:老撾萬榮一對情侶失蹤苇侵,失蹤者是張志新(化名)和其女友劉穎赶盔,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體榆浓,經(jīng)...
    沈念sama閱讀 46,646評論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡于未,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,709評論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了陡鹃。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片烘浦。...
    茶點(diǎn)故事閱讀 40,861評論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖杉适,靈堂內(nèi)的尸體忽然破棺而出谎倔,到底是詐尸還是另有隱情柳击,我是刑警寧澤猿推,帶...
    沈念sama閱讀 36,527評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響蹬叭,放射性物質(zhì)發(fā)生泄漏藕咏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,196評論 3 336
  • 文/蒙蒙 一秽五、第九天 我趴在偏房一處隱蔽的房頂上張望孽查。 院中可真熱鬧,春花似錦坦喘、人聲如沸盲再。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽答朋。三九已至,卻和暖如春棠笑,著一層夾襖步出監(jiān)牢的瞬間梦碗,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評論 1 274
  • 我被黑心中介騙來泰國打工蓖救, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留洪规,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,287評論 3 379
  • 正文 我出身青樓循捺,卻偏偏與公主長得像斩例,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子从橘,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,860評論 2 361

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