小技巧: 當ACE_Svc_Handler關閉時使用的默認行為
Stone Jiang
上一篇我們看了ACE_Svc_Handler::open()掛勾函數提供默認行為的一些技巧。因為在大多數情況下,它完成了服務創建時所需的所有事情:為輸入事件注冊新處理器和返回。
在ACE_Svc_Handler中關閉操作的默認行為要比在open()中初始化的代碼更為復雜。這是因為關閉時的掛勾函數包含了反應式(reactive)關閉和主動式(active-object)關閉兩種情
況,它們有相同的效果:刪除所有在反應器注冊的事件和確保刪除ACE_Svc_handler的派生類的對象。
?
反應式關閉(handle_close()): 當以下形況發生時被調用:
?1) 事件回調函數(譯注:handle_xxxx())返回-1時,或
?2) 調用ACE_Reactor::remove_handler(),傳入的標記不含 DONT_CALL時
?反應器框架將調用handle_close()掛勾函數。 ACE_Svc_handler::handle_close()的默認行為是調用ACE_Svc_Handler::destroy()銷毀事件處理器。
主動對象式關閉(close()): 當ACE_Svc_Handler 派生的對象通過activate()函數轉變為主動對象,處理器的的svc()方法在它產生的線程中運行后進行回調。當svc()返回時,線程
退出,但在它要退出時,仍在將要退出線程的上下文中,ACE_Task框架調用它的close()方法。ACE_Svc_Handler::close()方法調用handle_close()完成服務處理器的清理工作。
因此,不管ACE_Svc_Handler對象是反應式還是主動式,結束時的清理工作都在相同的地方:ACE_Svc_Handler::destroy()。如果對象是動態分配且不是ACE_Stream框架中的組成部
分,destroy()將刪除此對象,如果對象不是動態分配的,析構它的責任是創建時對象所在的封閉范圍。如果對象是ACE_Stream的模塊的組成部分,流和/或控制流的代碼負責管理
對象的生命期。destroy()函數應遵守這樣的規則來避免出現資源泄漏。
無論ACE_Svc_Handler對象是反應式銷毀還是主動式銷毀,都會引起相同的事情發生:~ACE_Svc_Handler() (處理器的析構函數)調用ACE_Svc_Handler::shutdown()函數完成服務處
理器的清理工作。shutdown()執行的這些清理操作是:
?
如果處理器與反應器是關聯的:
?? 取消處理器關聯的所有定時器。
?? 為處理器的流對象(如socket)刪除所有已注冊的事件。
如果處理器與再生器(recycler)是關系的,從再生器中清除處理器。
關閉流對象。
因此,我們看到的絕大多數情況服務處理器通過shutdown需要通過框架來清理工作,都是簡單的允許缺省的掛勾函數得以被調用。如果你的應用程序需求更多的關閉/清理規則,推
薦的地方是你的處理器的handle_close()掛勾方法。僅僅是確保把ACE_Svc_Handler::destroy()函數人作為你的handle_close()的最后一個動作,與框架清理行為的余下的行為合
并。
?