精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

C++的全鏈路追蹤方案,稍微有點高端

開發 后端
本文的知識點還是值得了解一下的,大家或許會用得到。在研究的過程中我也發現了一個基于此種方案的開源項目(call-stack-logger),感興趣的也可以去了解了解。

背景:本人主要在做C++ SDK的開發,需要給到業務端去集成,在集成的過程中可能會出現某些功能性bug,即沒有得到想要的結果。那怎么調試?

分析:這種問題其實調試起來稍微有點困難,它不像crash,當發生crash時還能拿到堆棧信息去分析,然而功能性bug沒有crash,也就沒法捕捉對應到當時的堆棧信息。因為不是在本地,也沒法用編譯器debug。那思路就剩log了,一種方式是考慮在SDK內部的關鍵路徑下打印詳細的log,當出現問題時拿到log去分析。然而總有漏的時候,誰能保證log一定打的很全面,很有可能問題就出現在沒有log的函數中。

解決:基于上面的背景和問題分析,考慮是否能做一個全鏈路追蹤的方案,把打印出整個SDK的調用路徑,從哪個函數進入,從哪個函數退出等。

想法1:可以考慮在SDK的每個接口都加一個context結構體參數,記錄下來函數的調用路徑,這可能是比較通用有效的方案,但是SDK接口已經固定了,更改接口要面臨的困難很大,業務端基本不會同意,所以這種方案不適合我們現有情況,當然一個從0開始建設的中間件和SDK可以考慮考慮。

想法2:有沒有一種不用改接口,還能追蹤到函數調用路徑的方案?

繼續沿著這個思路繼續調研,我找到了gcc和clang編譯器的一個編譯參數:-finstrument-functions,編譯時添加此參數會在函數的入口和出口處觸發一個固定的回調函數,即:

  1. __cyg_profile_func_enter(void *callee, void *caller); 
  2. __cyg_profile_func_exit(void *callee, void *caller); 

參數就是callee和caller的地址,那怎么將地址解析成對應函數名?可以使用dladdr函數:

  1. int dladdr(const void *addr, Dl_info *info); 

看下下面的代碼:

  1. // tracing.cc 
  2.  
  3. #include <cxxabi.h> 
  4. #include <dlfcn.h>  // for dladdr 
  5. #include <stdio.h> 
  6. #include <stdlib.h> 
  7. #include <string.h> 
  8.  
  9. #ifndef NO_INSTRUMENT 
  10. #define NO_INSTRUMENT __attribute__((no_instrument_function)) 
  11. #endif 
  12.  
  13. extern "C" __attribute__((no_instrument_function)) void __cyg_profile_func_enter(void *callee, void *caller) { 
  14.     Dl_info info; 
  15.     if (dladdr(callee, &info)) { 
  16.         int status; 
  17.         const char *name
  18.         char *demangled = abi::__cxa_demangle(info.dli_sname, NULL, 0, &status); 
  19.         if (status == 0) { 
  20.             name = demangled ? demangled : "[not demangled]"
  21.         } else { 
  22.             name = info.dli_sname ? info.dli_sname : "[no dli_sname nd std]"
  23.         } 
  24.  
  25.         printf("enter %s (%s)\n"name, info.dli_fname); 
  26.  
  27.         if (demangled) { 
  28.             free(demangled); 
  29.             demangled = NULL
  30.         } 
  31.     } 
  32.  
  33. extern "C" __attribute__((no_instrument_function)) void __cyg_profile_func_exit(void *callee, void *caller) { 
  34.     Dl_info info; 
  35.     if (dladdr(callee, &info)) { 
  36.         int status; 
  37.         const char *name
  38.         char *demangled = abi::__cxa_demangle(info.dli_sname, NULL, 0, &status); 
  39.         if (status == 0) { 
  40.             name = demangled ? demangled : "[not demangled]"
  41.         } else { 
  42.             name = info.dli_sname ? info.dli_sname : "[no dli_sname and std]"
  43.         } 
  44.         printf("exit %s (%s)\n"name, info.dli_fname); 
  45.  
  46.         if (demangled) { 
  47.             free((void *)demangled); 
  48.             demangled = NULL
  49.         } 
  50.     } 

這是測試文件:

  1. // test_trace.cc 
  2. void func1() {} 
  3.  
  4. void func() { func1(); } 
  5.  
  6. int main() { func(); } 
  7. 將test_trace.cc和tracing.cc文件同時編譯鏈接,即可達到鏈路追蹤的目的: 
  8. g++ test_trace.cc tracing.cc -std=c++14 -finstrument-functions -rdynamic -ldl;./a.out 
  9. 輸出:enter main (./a.out
  10. enter func() (./a.out
  11. enter func1() (./a.out
  12. exit func1() (./a.out
  13. exit func() (./a.out
  14. exit main (./a.out

如果在func()中調用了一些其他的函數呢?

  1. #include <iostream> 
  2. #include <vector> 
  3.  
  4. void func1() {} 
  5.  
  6. void func() { 
  7.     std::vector<int> v{1, 2, 3}; 
  8.     std::cout << v.size(); 
  9.     func1(); 
  10.  
  11. int main() { func(); } 

再重新編譯后輸出會是這樣:

  1. enter [no dli_sname nd std] (./a.out
  2. enter [no dli_sname nd std] (./a.out
  3. exit [no dli_sname and std] (./a.out
  4. exit [no dli_sname and std] (./a.out
  5. enter main (./a.out
  6. enter func() (./a.out
  7. enter std::allocator<int>::allocator() (./a.out
  8. enter __gnu_cxx::new_allocator<int>::new_allocator() (./a.out
  9. exit __gnu_cxx::new_allocator<int>::new_allocator() (./a.out
  10. exit std::allocator<int>::allocator() (./a.out
  11. enter std::vector<int, std::allocator<int> >::vector(std::initializer_list<int>, std::allocator<int> const&) (./a.out
  12. enter std::_Vector_base<int, std::allocator<int> >::_Vector_base(std::allocator<int> const&) (./a.out
  13. enter std::_Vector_base<int, std::allocator<int> >::_Vector_impl::_Vector_impl(std::allocator<int> const&) (./a.out
  14. enter std::allocator<int>::allocator(std::allocator<int> const&) (./a.out
  15. enter __gnu_cxx::new_allocator<int>::new_allocator(__gnu_cxx::new_allocator<int> const&) (./a.out
  16. exit __gnu_cxx::new_allocator<int>::new_allocator(__gnu_cxx::new_allocator<int> const&) (./a.out
  17. exit std::allocator<int>::allocator(std::allocator<int> const&) (./a.out
  18. exit std::_Vector_base<int, std::allocator<int> >::_Vector_impl::_Vector_impl(std::allocator<int> const&) (./a.out
  19. exit std::_Vector_base<int, std::allocator<int> >::_Vector_base(std::allocator<int> const&) (./a.out

上面我只貼出了部分信息,這顯然不是我們想要的,我們只想要顯示自定義的函數調用路徑,其他的都想要過濾掉,怎么辦?

這里可以將自定義的函數都加一個統一的前綴,在打印時只打印含有前綴的符號,這種個人認為是比較通用的方案。

下面是我過濾掉std和gnu子串的代碼:

  1. if (!strcasestr(name"std") && !strcasestr(name"gnu")) { 
  2.     printf("enter %s (%s)\n"name, info.dli_fname); 
  3.  
  4. if (!strcasestr(name"std") && !strcasestr(name"gnu")) { 
  5.     printf("exit %s (%s)\n"name, info.dli_fname); 

重新編譯后就會輸出我想要的結果:

  1. g++ test_trace.cc tracing.cc -std=c++14 -finstrument-functions -rdynamic -ldl;./a.out 
  2. 輸出:enter main (./a.out
  3. enter func() (./a.out
  4. enter func1() (./a.out
  5. exit func1() (./a.out
  6. exit func() (./a.out
  7. exit main (./a.out

還有一種方式是在編譯時使用下面的參數:

  1. -finstrument-functions-exclude-file-list 

它可以排除不想要做trace的文件,但是這個參數只在gcc中可用,在clang中卻不支持 ,所以上面的字符串過濾方式更通用一些。

上面只能拿到函數的名字,不能定位到具體的文件和行號,如果想要獲得更多信息,需要結合bfd系列參數(bfd_find_nearest_line)和libunwind一起使用,大家可以繼續研究。。。

tips1:這是一篇拋磚引玉的文章,本人不是后端開發,據我所知后端C++中有很多成熟的trace方案,大家有更好的方案可以留言,分享一波。

tips2:上面的方案可以達到鏈路追蹤的目的,但本人最后沒有應用到項目中,因為本人在做的項目對性能要求較高,使用此種方案會使整個SDK性能下降嚴重,無法滿足需求正常運行。于是暫時放棄了鏈路追蹤的這個想法。

本文的知識點還是值得了解一下的,大家或許會用得到。在研究的過程中我也發現了一個基于此種方案的開源項目(call-stack-logger),感興趣的也可以去了解了解。

 

責任編輯:武曉燕 來源: 程序喵大人
相關推薦

2023-10-16 23:43:52

云原生可觀測性

2023-01-30 22:34:44

Node.js前端

2022-07-22 07:59:17

日志方案

2022-05-23 08:23:24

鏈路追蹤SleuthSpring

2025-10-10 08:58:13

2022-05-19 13:33:39

系統客戶端鏈路追蹤

2024-12-16 13:34:35

2022-05-25 08:23:32

ZipKinTwitter開源項目

2025-03-11 14:16:09

2025-05-26 08:50:00

SLF4JMDC全鏈路追蹤

2024-09-06 12:24:19

2025-01-20 08:10:00

微服務架構SLF4J

2021-11-18 10:01:00

Istio 全鏈路灰度微服務框架

2022-12-05 19:15:12

得物云原生全鏈路

2018-07-03 15:56:59

騰訊

2022-09-15 10:03:42

Jaeger分布式追蹤系統

2023-08-24 22:13:31

2020-12-16 09:24:18

Skywalking分布式鏈路追蹤

2024-06-07 13:04:31

2024-01-05 00:29:36

全鏈路灰度發布云原生
點贊
收藏

51CTO技術棧公眾號

亚洲性xxxx| 欧美午夜无遮挡| 成人三级在线| 人人妻人人爽人人澡人人精品| 精品国产乱码久久久久久果冻传媒| 精品婷婷伊人一区三区三| 日本黄色a视频| 亚洲av综合色区无码一区爱av| 新67194成人永久网站| 久久精品福利视频| 女同性恋一区二区三区| 电影亚洲一区| 亚洲一区二区三区视频在线播放| 鲁丝一区二区三区免费| 精品人妻少妇嫩草av无码专区| 99热这里只有成人精品国产| 日韩视频亚洲视频| 野外性满足hd| 77成人影视| 欧美三级蜜桃2在线观看| 草草视频在线免费观看| 青青青青在线| 久久综合给合久久狠狠狠97色69| 亚洲综合视频1区| 欧美一级黄视频| 宅男噜噜噜66国产日韩在线观看| 久久成人一区二区| 成年人在线免费看片| 国产欧美自拍一区| 91麻豆精品国产91久久久久久 | 精品国精品自拍自在线| 午夜免费看视频| 永久免费毛片在线播放| 亚洲自拍偷拍av| 在线观看福利一区| 成人在线观看黄色| 久久精品视频免费| 久久精品人成| 特黄视频在线观看| 国产白丝网站精品污在线入口| 国产女人18毛片水18精品| 波多野结衣一区二区三区在线| 国产视频一区三区| 国内自拍欧美激情| 中文在线观看免费网站| 亚洲欧美网站在线观看| 日韩中文字幕网站| 91导航在线观看| 极品美女一区二区三区| 亚洲免费小视频| 亚洲专区区免费| 日韩精品欧美大片| 亚洲精品美女在线观看| www国产视频| 欧美一区二区三区红桃小说| 亚洲国产成人爱av在线播放| 欧美激情 亚洲| 国产精品欧美大片| 日韩av综合网站| 欧美大片免费播放器| 四虎5151久久欧美毛片| 日韩精品视频在线观看网址| 在线观看国产网站| 国产成人黄色| 中文字幕日韩免费视频| 91精品久久久久久久久久久久| av中文一区| 色婷婷av一区二区三区久久| 亚洲 欧美 国产 另类| 99久久久久国产精品| 欧美成人精品在线| 国产亚洲精品成人| 鲁大师成人一区二区三区| 国产精彩精品视频| 一区二区三区精彩视频| 国产麻豆精品久久一二三| 国产99视频精品免费视频36| 天天操天天操天天操| 久久亚洲二区三区| 亚洲第一页在线视频| 最新黄网在线观看| 精品成人乱色一区二区| 超碰av在线免费观看| 日韩专区视频| 精品毛片乱码1区2区3区| 亚洲制服丝袜在线播放| 成人情趣视频| 欧美激情二区三区| 91丝袜一区二区三区| 麻豆国产精品视频| 国产久一道中文一区| 国产九九在线| 亚洲综合在线免费观看| 免费在线观看日韩视频| 成人污污www网站免费丝瓜| 亚洲成人黄色在线| 成人一级片免费看| 99国产精品视频免费观看一公开| 国产精品精品国产| 懂色av蜜臀av粉嫩av分享吧| 91社区在线播放| 97精品国产97久久久久久粉红| av影院在线免费观看| 欧美日韩一区 二区 三区 久久精品| 亚洲成a人无码| 欧美一级精品片在线看| 欧美激情在线一区| 亚洲在线视频播放| 久久日韩粉嫩一区二区三区| 精品一区二区三区毛片| 欧美精品高清| 亚洲国产精品女人久久久| 国产在线观看免费视频软件| 久久久久欧美精品| dy888夜精品国产专区| 成人免费在线视频网| 婷婷激情综合网| 色婷婷综合在线观看| 精品国产成人| 91精品国产高清久久久久久久久| 国产精品视频一二区| 久久九九全国免费| 黄页网站在线观看视频| 欧一区二区三区| 色七七影院综合| 高潮毛片又色又爽免费 | www.色多多| 精品av久久久久电影| 92国产精品视频| 免费黄色在线观看| 欧美色大人视频| 老牛影视av老牛影视av| 国产精品呻吟| 国产一区免费在线观看| 国内在线视频| 欧美不卡一区二区三区四区| 三级影片在线看| 久久成人免费网站| 中文字幕乱码一区二区三区| 亚洲天堂1区| 一区二区三区动漫| 久久久精品毛片| 国产女主播一区| 亚洲五月天综合| 少妇精品久久久一区二区| 欧美自拍视频在线| 免费理论片在线观看播放老| 欧美日韩亚洲一区二| 亚洲第九十七页| 国产日韩高清一区二区三区在线| 精品视频一区二区| 国产一二在线播放| 国产视频精品自拍| 一级片免费在线播放| 国产日韩av一区二区| 日本成人中文字幕在线| 欧美日韩在线观看视频小说| 国产精品成人v| 秋霞成人影院| 欧美一级二级三级乱码| 国产一级性生活| 91最新地址在线播放| 99999精品视频| 国产精品一在线观看| 国产精品欧美日韩一区二区| 1024免费在线视频| 欧美疯狂性受xxxxx喷水图片| www.99re6| 国产精品系列在线播放| 少妇人妻无码专区视频| 伊人成综合网yiren22| 国产精品色婷婷视频| 国产淫片在线观看| 欧美精品一区二区三区四区| 秋霞精品一区二区三区| 国产精品另类一区| 四虎国产精品永久免费观看视频| 亚洲黄色一区| 日本一区免费| 欧美一区一区| 欧美综合一区第一页| 亚洲搞黄视频| 亚洲国产精品免费| wwwwww在线观看| 亚洲精品乱码久久久久久黑人 | 99热这里只有精| 亚洲成年人网站在线观看| 国产美女精品久久| 国产一区在线观看麻豆| 777精品久无码人妻蜜桃| 全球成人免费直播| 国产精品 日韩| 国产黄色一区| 91极品女神在线| 巨大荫蒂视频欧美另类大| 亚洲激情中文字幕| ,一级淫片a看免费| 精品国产福利在线| 伊人在线视频观看| 国产亚洲精品久| 绯色av蜜臀vs少妇| 免费在线观看成人| 国产不卡一区二区视频| 欧美成人激情| 欧美一区二区福利| 99热这里只有精品首页| 成人h视频在线| 国模冰冰炮一区二区| 欧美猛男性生活免费| 国产一二三区在线| 日韩电影免费观看在线观看| av中文在线观看| 欧美艳星brazzers| 亚洲久久在线观看| 亚洲图片欧美一区| 成人性生活毛片| 中文字幕欧美区| 野外性满足hd| 97se狠狠狠综合亚洲狠狠| 久久久久久无码精品人妻一区二区| 另类天堂av| 国产乱子伦农村叉叉叉| 欧美国产综合| 只有这里有精品| 国产精品99一区二区三| 日本一区二区三区在线视频 | 国产河南妇女毛片精品久久久| 手机在线免费观看毛片| 先锋影音久久久| 亚洲熟妇无码另类久久久| 欧美午夜久久| japanese在线播放| 午夜国产一区二区| 色99中文字幕| 欧美亚洲国产精品久久| 日日噜噜噜噜夜夜爽亚洲精品| 网曝91综合精品门事件在线| 国产欧美综合精品一区二区| 高清日韩欧美| 不卡视频一区二区| 超碰97久久国产精品牛牛| 99re国产| 中文字幕一区日韩精品| 2014国产精品| 亚洲3区在线| 国产高清自拍99| 91欧美日韩在线| 国产精品高清一区二区三区| jizz国产精品| 精品乱色一区二区中文字幕| 网友自拍一区| 欧美一区视久久| 日韩系列欧美系列| 国产精品夜夜夜爽张柏芝| 一区二区三区午夜探花| 国产制服91一区二区三区制服| 综合激情视频| 免费一级特黄毛片| 午夜在线一区二区| 亚洲一级免费观看| 国产米奇在线777精品观看| 亚洲精品一区二区18漫画| 国产a区久久久| 网站免费在线观看| 中文字幕国产一区| 老湿机69福利| 黄色成人在线免费| 狠狠狠狠狠狠狠| 欧美一区二区私人影院日本| 成人午夜视频一区二区播放| 亚洲国产精品久久久| 国产在线观看黄| 久久天堂电影网| 黄色在线网站噜噜噜| 国产精品激情自拍| 影音先锋欧美激情| 日韩啊v在线| 欧美精品导航| 欧美 日韩精品| 久久69国产一区二区蜜臀| 久草视频福利在线| 久久精品视频免费| 欧美交换国产一区内射| 色哟哟在线观看一区二区三区| 亚洲熟妇无码久久精品| 精品福利一区二区三区免费视频| 免费国产在线视频| 九九久久久久99精品| 亚洲私拍视频| 亚洲va电影大全| 亚洲成a人片77777在线播放| 樱花www成人免费视频| 999亚洲国产精| 福利视频999| 久久免费美女视频| 欧美成人三级在线观看| 欧美亚洲动漫另类| 亚洲国产福利视频| 国产一区二区三区视频在线观看| 欧美野外wwwxxx| 国产精品欧美风情| 日韩有码av| 黄色一级片国产| 免费日本视频一区| www.超碰97| 一区二区高清视频在线观看| 亚洲大尺度在线观看| 亚洲高清在线观看| 污影院在线观看| 成人夜晚看av| 国产91一区| 自慰无码一区二区三区| 国产一区二区精品在线观看| a天堂中文字幕| 精品国产老师黑色丝袜高跟鞋| 国产成人精品无码高潮| 中文字幕日韩电影| 日韩另类视频| 欧美亚洲免费在线| 一本色道久久综合一区 | 久久综合久色欧美综合狠狠| 久久久国产精品黄毛片| 9191成人精品久久| av资源种子在线观看| 日本久久久久亚洲中字幕| 成人三级av在线| 一本大道东京热无码aⅴ| 久久精品国产色蜜蜜麻豆| 在线观看免费小视频| 日韩欧美中文免费| 少妇性bbb搡bbb爽爽爽欧美| 久久久久久久成人| 99精品中文字幕在线不卡| 99亚洲国产精品| 国产91露脸合集magnet| 欧美黑吊大战白妞| 欧美一级一区二区| 亚洲婷婷噜噜| 成人影片在线播放| 精品69视频一区二区三区Q| 欧洲成人午夜精品无码区久久| 亚洲欧美日韩系列| 99草在线视频| 色综合久久精品亚洲国产| 日韩三级网址| 97免费视频观看| av在线不卡免费看| 成人免费视频毛片| 亚洲人成网7777777国产| 台湾佬成人网| 翔田千里亚洲一二三区| 九九视频精品免费| 内射一区二区三区| 日韩一级高清毛片| 蜜臀av国内免费精品久久久夜夜| 高清国产在线一区| 99精品国产99久久久久久福利| 国产一级伦理片| 色婷婷久久久久swag精品| 成人免费视频| 亚洲va欧美va国产综合剧情| 欧美日韩专区| 亚洲久久久久久| 欧洲色大大久久| 免费的黄网站在线观看| 99久久久久国产精品免费| 亚洲精品欧洲| 国产美女免费无遮挡| 欧美精三区欧美精三区| 免费在线国产视频| 欧美极品一区二区| 久久草av在线| 毛片a片免费观看| 亚洲精品视频网上网址在线观看| 韩国精品主播一区二区在线观看 | 最近中文字幕免费mv2018在线| 成人av蜜桃| 亚洲综合二区| 日本 欧美 国产| 日韩高清免费观看| 日韩色性视频| 免费观看国产精品视频| 欧美国产欧美亚州国产日韩mv天天看完整| 亚洲性生活大片| 91精品国产91久久久| 日韩精品午夜| www男人天堂| 精品视频一区三区九区| 91精品国产黑色瑜伽裤| 天天综合色天天综合色hd| 国产91精品在线观看| 波多野结衣家庭主妇| 欧美激情aaaa| 99久久夜色精品国产亚洲96| 女同性恋一区二区三区| 在线不卡免费欧美| 91久久国产综合久久91猫猫| 国产精品久久成人免费观看| 久久久久久电影| 亚洲国产视频一区二区三区|