真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

作為.Net攻城師,所必需掌握的.NetProfiling技術(shù)

眾所周知,性能問題是所有實用應用在迭代過程中必然要面對的問題。對于此類問題,簡單地投入更多硬件資源的做法可能會取得一定效果。但總的來看,此類做法的邊際成本是不斷上升的。換言之,隨著性能需求的上漲,要換取同樣的性能提升,僅憑硬件升級所需要的成本會越來越高。故而性能優(yōu)化是每一位運維/軟件開發(fā)人員必須掌握的技術(shù)。

十年專注成都網(wǎng)站制作,企業(yè)網(wǎng)站制作,個人網(wǎng)站制作服務,為大家分享網(wǎng)站制作知識、方案,網(wǎng)站設計流程、步驟,成功服務上千家企業(yè)。為您提供網(wǎng)站建設,網(wǎng)站制作,網(wǎng)頁設計及定制高端網(wǎng)站建設服務,專注于企業(yè)網(wǎng)站制作,高端網(wǎng)頁制作,對成都門簾等多個方面,擁有多年的網(wǎng)站設計經(jīng)驗。

.Net Profiling

在進行應用性能優(yōu)化實踐時,首先面對的就是熱點定位,即確定那些帶來巨大資源耗散的代碼位置。而在不借助外部工具的前提下,定位資源熱點是一件相當困難的事。它需要當事人對于應用實現(xiàn)本身有一個整體的把握,了解應用架構(gòu)內(nèi)每一個功能模塊代碼的路徑與細節(jié)。與此同時,當事人還要對于應用實現(xiàn)所依賴的第三方功能庫的表現(xiàn)有一定的把握。對于那些具備一定規(guī)模的應用系統(tǒng),具備前述素質(zhì)的工程師的數(shù)量屈指可數(shù)。而即便是這些百里挑一的優(yōu)秀人才,其熱點預估也不能保證一定是準確的。

Profiling 技術(shù) 的提出正是為了解決熱點定位而提出的,它以程序的實際運行數(shù)據(jù)來幫助工程師們來理解應用行為,極大地簡化了工程師們的工作。而對于像 .Net 這樣比較成熟的技術(shù)棧,專家們(如 Bill Chiles 和 趙頡)都推薦工程人員通過實際的 Profiling 數(shù)據(jù)來定位性能瓶頸。實際操作中,獲取 .Net Profiling 數(shù)據(jù)的功能被實現(xiàn)為一個被稱為 Profiler 的 COM 組件實體。下文中,我們將就 Profiler 本身的實現(xiàn)進行一些探討。

Profiling API

.Net Profiler 在本質(zhì)上是 CLR 的一個插件,它通過應用 Profiling API 來保持與 CLR 的信息溝通,并以此獲取到 .Net 應用的運行時數(shù)據(jù)。通常來說,Profiler 的實體都表現(xiàn)為一個動態(tài)鏈接庫(即 .dll 文件),CLR 在運行時會去加載該庫(CLR 加載 Profiler 的詳細配置可以參考 這篇文檔),并在程序運行的特定階段向庫發(fā)送信息并接受庫所返回的信息。

需要特別強調(diào)的一點是,雖然被稱作是 Profiling API,但 CLR 的這套接口能做的可不僅僅是簡單地度量應用的運行時間和內(nèi)存耗散。實踐上,profiling API 能夠完成諸如代碼覆蓋、運行時插入等許多高級功能。不過,正如 MSDN Profiling 綜述文檔 所強調(diào)的,Profiling 對于應用本身應該透明。也就是說,在編寫應用的時候,開發(fā)人員不應該在自己的邏輯中依賴 Profiler 或者被其影響。

對于 .Net 技術(shù)棧而言,由于環(huán)境本身引入了 application domain, GC, managed exception handling, JIT 等高級特性,Profiling 所展現(xiàn)的就不能僅止于應用運行所消耗的時間或者內(nèi)存。為了能夠真正地表現(xiàn)出運行時行為,Profiling API 中提供了包含這些特性的數(shù)據(jù)的接口。這就使得 Profiling API 在設計并不是如很多人想得那樣直觀。

常見的 .Net Profiler 實現(xiàn)多采取如下架構(gòu): 
作為 .Net 攻城師,所必需掌握的 .Net Profiling 技術(shù) 
.Net Profiler 架構(gòu)

其中,ICorProfilerCallback 與 ICorProfilerInfo 就是 Profiling API 中最常被應用到的兩個。在應用運行時,Profiler DLL 會被加載到應用所在的進程中。通過實現(xiàn) ICorProfilerCallback 下特定功能的接口,Profiler DLL 會在應用運行時收到相關的動作執(zhí)行通知。例如,如果在 Profiler DLL 中實現(xiàn)了 ICorProfilerCallback::AssemblyLoadFinished 接口,那么在應用運行中每加載完一個程序集 時,Profiler DLL 中該接口的實現(xiàn)代碼就會被調(diào)用。與此類似,Profiler DLL 可以靠實現(xiàn) ICorProfilerInfo 下的接口來完成對被監(jiān)測應用狀態(tài)的獲取。

需要補充說明一點,上文所說的 ICorProfilerCallback 接口實際上存在有 ICorProfilerCallback ~ ICorProfilerCallback7 這樣7個版本的接口定義。高標號的接口版本向下兼容,但會提供新的功能擴展。不過,更高標號的接口往往也需要有更新版本的 CLR 來支持(如調(diào)用 ICorProfilerCallback7 需要在環(huán)境中部署 .Net Framework 4.6.1 以上版本),在實際使用時需要多加注意。

目前,Profiling API 可以被任何非托管的 COM 兼容的語言所調(diào)用。另外,API 本身的實現(xiàn)非常高效的,不會帶來大到足以導致 profiling 失效的額外性能負擔。也正因此,基于 Profiling API 完全可以實現(xiàn)一個抽樣 profiler(對于 profiling 模式的探討可參照 這篇文獻 的 2.1 章節(jié)內(nèi)容)。

目前 Profiling API 所支持的特性

正如前文所述,Profiler 對于程序行為的描述源自 profiling API 所提供的信息。在目前版本中,憑借 profiling API 能夠獲取到下列事件的消息通知:

  1. CLR 的啟動與關停

  2. application domain 的創(chuàng)建與關閉

  3. 程序集的加載與卸載

  4. 模塊(Module)的加載與卸載

  5. COM vtable 的創(chuàng)建與銷毀

  6. JIT 編譯與 code-pitching 的出發(fā)

  7. 類的加載卸載

  8. 線程的創(chuàng)建與銷毀

  9. 函數(shù)的進入與返回

  10. 托管代碼與非托管代碼的執(zhí)行切換

  11. 運行時掛起

  12. 運行時堆內(nèi)存信息與 GC 活動

隨著 .Net 技術(shù)的演進,未來的 Profiling API 或許能夠提供更多的信息。不過,以下功能點是 Profiling API 不會實現(xiàn)的,請在應用時回避:

  1. 非托管代碼的執(zhí)行信息

  2. 運行時修改自身代碼的應用的 Profiling(如 AOP)

  3. 邊界檢驗

  4. 遠程 profiling

  5. 高可靠性環(huán)境下的 profiling

線程相關

對于加載了 Profiler DLL 的進程而言,其在創(chuàng)建新線程時,新線程本身也會產(chǎn)生 ICorProfilerCallback 接口下定義的各種事件通知。這一過程中,Profiler 不必去顯式地指定一個 ThreadID 以使得 Profiling API 生效。同樣的,Profiler 完全可以簡單地在代碼中使用 thread-local 的存儲方式,用不著費心地去進行存儲位置的全局重定向。

當然,還是有一些要點需要我們在并發(fā)背景下留意。例如,Profiling API 本身并不能保障數(shù)據(jù)結(jié)構(gòu)的線程安全,因此我們需要在可能產(chǎn)生并行訪問的地方給 Profiler 代碼沖突區(qū)加鎖以保證 Profiler 的行為符合預期。同樣,對于多線程的應用場景來說,Profiler 不應該假設 ICorProfilerCallback 的各個接口存在一定的先后順序。舉例來說,一個有兩個同樣線程的程序在運行時可能會先產(chǎn)生一個 FunctionEnter 然后才產(chǎn)生 ICorProfilerCallback::JITCompilationFinished。

還有一個線程相關的問題是來自于 COM 接口的。上文中我們說過 Profiler 事實上是實現(xiàn)為一個 COM 組件的,但其實 CLR 在運行時并不會去初始化 COM。這是為了避免在應用代碼指定線程模型前,CLR 調(diào)用 [CoInitialize][Ref12] 來指定應用線程模型。同樣地,在 Profiler 內(nèi)部,不要去調(diào)用 CoInitialize 以避免與應用代碼產(chǎn)生沖突。

調(diào)用棧

獲取調(diào)用棧信息是應用 Profiling 時的一項關鍵需求。針對這一需求,Profiling API 提供給 Profiler 編寫者兩種實現(xiàn)方式:??煺蘸偷褂皸?。

棧快照是在特定時刻對特定線程調(diào)用棧的一次追蹤。需要注意,Profiling API 僅支持對棧上托管函數(shù)的追蹤。如果 Profiler 需要追蹤棧上的非托管函數(shù),則需要自己提供一個棧遍歷器出來。讀者如對 Profiler ??煺諜C制的實現(xiàn)詳情感興趣,可以參照 Divid Broman 的 這篇博客 內(nèi)容。

頻繁使用??煺諘?CLR 帶來過多的額外性能損耗。因此,如果需要頻繁進行棧追蹤,那么 Profiler 應該通過 FunctionEnter2, FunctionLeave2, FunctionTailCall2 及 ICorProfilerCallback::Exception* 等一系列接口構(gòu)建出當前應用調(diào)用棧的一個倒影。如此,Profiler 即可低消耗地進行棧追蹤操作。

其他需要留意之處

前文強調(diào)過,Profiler 是一個非托管的 DLL 庫,會在應用運行時被加載到 CLR 中并與應用處于同一進程空間下。如此,Profiler DLL 實質(zhì)上是不受托管代碼的訪問控制的。其運行唯一的限制就是運行 Profiler 的 OS 用戶必須擁有足夠權(quán)限。因此,對于要部署 Profiler 的技術(shù)人員來說,必須要明白這其中可能的風險,提早進行準備。例如可以把 Profiler DLL 加到訪問控制列表(ACL)中以免惡意用戶對其加以利用。

還有,Profiler DLL 作為 CLR 的一個插件,其運行錯誤可能會引起 CLR 本身的崩潰,在實施時一定要足夠小心。而對于那些運行在棧內(nèi)存空間緊張的環(huán)境下的 Profiler,要警惕因為 ICorProfilerCallback 導致棧溢出而引起的應用崩潰。在這種資源受限的環(huán)境中,要盡可能地減少 Profiler 自身的資源耗散。盡力避免原本能夠運行的應用因為 Profiler 而導致無法運行的情況。

結(jié)語

本文簡述了 .Net Profiling 技術(shù)的總體情況,并就其中的一些重要技術(shù)點進行了闡述。希望能幫助讀者初步理解 .Net Profiling 技術(shù)。后續(xù),我們將具體到代碼實施層面,就 .Net Profiling 的實現(xiàn)進行詳細討論

OneAPM 助您輕松鎖定 .NET 應用性能瓶頸,通過強大的 Trace 記錄逐層分析,直至鎖定行級問題代碼。以用戶角度展示系統(tǒng)響應速度,以地域和瀏覽器維度統(tǒng)計用戶使用情況。想閱讀更多技術(shù)文章,請訪問 OneAPM 官方博客。 
本文轉(zhuǎn)自 OneAPM 官方博客


名稱欄目:作為.Net攻城師,所必需掌握的.NetProfiling技術(shù)
鏈接URL:http://weahome.cn/article/ieisdo.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部