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

一個耗時4小時的內存泄漏問題

開發 后端
首先確定內存泄漏問題出現的時間,發現在該時間點的上線有兩次代碼提交,其中一個就是我的。于是立刻排查這兩次代碼的改動,確定了另一個同事的代碼不可能會有內存問題后我知道肯定是自己的代碼出現了問題。

上周像往常一樣例行檢查線上機器性能,突然發現一個服務的內存使用率是這樣的:

很顯然該服務存在內存泄漏問題,趕緊排查問題。

問題排查

首先確定內存泄漏問題出現的時間,發現在該時間點的上線有兩次代碼提交,其中一個就是我的。于是立刻排查這兩次代碼的改動,確定了另一個同事的代碼不可能會有內存問題后(因為另一個同事的上線僅僅修改了配置)我知道肯定是自己的代碼出現了問題。

確定了問題所在后趕緊把自己的代碼回滾掉,接下來就可以放心debug了。

Debug

什么是內存泄漏?

簡單的講就是程序員申請的內存在使用完后沒有還給操作系統,由于筆者使用的是C++語言,因此內存泄漏一般是這樣的: 

  1. obj* o = new obj();  
  2. ...  
  3. // 使用完obj后沒有delete掉 

肯定有什么地方申請了內存后沒有調用delete釋放內存。

在這里介紹一下筆者的代碼改動,我的任務其實是重構一段代碼,把這段代碼并行化。也就是舊的邏輯是在一個線程中串行執行的,現在我要把這段邏輯放到兩個線程中并行執行,這是最讓人頭疼的任務之一,并行化改造是比較容易出bug的。

接下來梳理了一遍中所有內存的申請和釋放,這其中包括:

  •  使用new/delete分配釋放的內存
  •  使用內存池分配釋放的內存

仔細梳理一遍后沒有發現任何問題,該釋放的內存都已經釋放掉了,這時筆者已經開始懷疑人生了 :) ,很顯然還有一段沒有注意到的地方出現了問題,這是必然的,雖然知道問題必然出現在改動的這些代碼里但是我并不能確定出現的位置。

沒有辦法,到這里基本上已經要放棄自己人肉debug了,想利用一些內存檢測工具來幫助自己確定問題。

常見的內存泄漏檢測工具包括valgrind、gperftools等,valgrind的好處在于無需重新編譯代碼即可進行內存檢測,但是缺點是會使得程序運行非常緩慢,官方文檔給的說法是會比正常的程序運行慢20-30倍;gperftools則需要重新編譯可執行程序。這些工具需要下載安裝測試,其中還涉及到申請機器權限等問題,筆者覺得還是比較麻煩,況且這個問題也不是大海撈針一樣,問題肯定出在了并行化的這段代碼中。

到這里我決定再換一個思路來排查問題,既然代碼重構后開始并行執行,那么出現問題大概率是因為多線程問題,遇到多線程問題首先重點排查的就是線程間的共享數據。

多線程問題的關鍵——共享數據

我們知道如果線程之間沒有共享數據那么就不會有線程安全問題,我們使用的鎖、信號量、條件變量等其實都是用來保護共享數據的,比如鎖通常是用來包括臨界區的,臨界區中的代碼操作的就是線程共享數據;信號量使用的一個經典場景就是生產者消費者問題,生產者線程以及消費者線程都會操作同一個隊列,這里的隊列就是共享數據。

沿著這個思路開始找在兩個線程中都使用到的共享數據,果不其然,在一個角落中發現了這樣一段代碼: 

  1. auto* pb = global->mutable_obj(); 

這是分配protobuf對象的一段代碼,protobuf是Google開發是一種類似于JSON、XML的技術,因此常用于網絡通信和數據交換等場景,比如RPC等。

如果你不了解protobuf也沒有關系,實際上上面的這段代碼的要做的事情是這樣的: 

  1. if (global->obj == NULL) {  
  2.   global->obj = new obj();  
  3.  
  4. return global->obj; 

值得注意的是這段代碼現在會在兩個線程中執行,顯然問題就出現在了這里。

那么問題是怎么出現的呢?

我們假設有兩個線程,線程A和線程B,當這樣一段代碼在線程AB中同時執行時可能會有以下場景:

  •  線程A拿到global->obj并檢測到此時的global->obj為空,因此決定為其分配內存,但不巧的是此時發生線程切換,線程A在為global->obj分配內存前被暫停運行,如下所示: 
  1. if (global->obj == NULL) {  
  2.     <------- 線程切換,線程A被暫停執行   
  3.     global->obj = new obj();  
  4.  
  5. return global->obj; 
  •  線程A被暫停運行后線程B開始執行,這段代碼同樣會在線程B中執行一遍,因此線程B會首先檢查global->obj發現為空,因此為global->obj分配內存,分配完內存后發生線程切換,線程B被暫停運行,如下所示: 
  1. if (global->obj == NULL) {  
  2.     global->obj = new obj();  
  3.     <------- 線程切換,線程B被暫停執行   
  4.  
  5. return global->obj; 
  • 線程B被暫停運行后調度器決定重新運行線程A,此時線程A開始從被中斷的地方繼續運行,還記得線程A是從哪里被中斷的嗎,沒錯,就是在為global->obj分配內存前被中斷的,此時線程A繼續運行,也就是說global->obj = new obj()這段代碼又被執行了一次,雖然線程B已經為global->obj分配了內存。

Oops,典型的內存泄漏,線程B分配的內存再也無法被正常釋放掉了。

至此,我們已經找到了問題的原因,罪魁禍首就是共享數據,關鍵的一點是要意識到你的線程會隨時被中斷執行,CPU會隨時切換到其它線程。

代碼修復也非常簡單,再新增一個變量,兩個線程不在使用共享數據,到這里問題就解決了,從發現問題到完成修復耗時大概4小時。

經驗教訓

代碼的并行化重構是一件非常棘手的任務,很容易出現線程安全問題,解決線程安全問題首先要考慮的不是要不要加鎖,而是多個線程是否真的有必要使用共享數據,沒有必要的話多個線程操作私有數據根本就不會出現線程安全問題。

當出現線程安全問題時,第一時間重點排查線程使用的共享數據。

內存泄漏檢測工具

雖然這些沒有使用檢測工具全靠人肉debug其實還是因為問題排查范圍比較小,如果我們根本就不知道問題出現在了那次代碼改動那么檢測工具就非常重要了,在這里簡單介紹一下valgrind的使用,詳細的介紹請參考官方文檔。

假設有這樣一段問題代碼: 

  1. #include <stdlib.h>  
  2. void f(void)    
  3.  
  4.    int* x = malloc(10 * sizeof(int));  
  5.    x[10] = 0;        // 問題1: 越界  
  6. }                    // 問題2: 內存泄漏,x沒有被釋放掉   
  7. int main()   
  8.  
  9.    f();  
  10.    return 0;  

這段代碼中有兩個問題:一個是數據的越界訪問;另一個是內存泄漏。將該程序編譯為myprog。

接下來使用valgrind來檢查該程序,使用以下命令: 

  1. valgrind --leak-check=yes myprog 

運行完成后valgrind會給出檢測報告,關于程序越界訪問會給出這樣的輸出: 

  1. ==19182== Invalid write of size 4  
  2. ==19182==    at 0x804838F: f (example.c:6)  
  3. ==19182==    by 0x80483AB: main (example.c:11)  
  4. ==19182==  Address 0x1BA45050 is 0 bytes after a block of size 40 alloc'd  
  5. ==19182==    at 0x1B8FF5CD: malloc (vg_replace_malloc.c:130) 
  6. ==19182==    by 0x8048385: f (example.c:5)  
  7. ==19182==    by 0x80483AB: main (example.c:11) 

第一行告訴你代碼中存在Invalid write,也就是無效的寫,并給出了問題出現的位置。

關于內存泄漏問題會給出這樣的輸出: 

  1. ==19182== 40 bytes in 1 blocks are definitely lost in loss record 1 of 1  
  2. ==19182==    at 0x1B8FF5CD: malloc (vg_replace_malloc.c:130)  
  3. ==19182==    by 0x8048385: f (example.c:5)  
  4. ==19182==    by 0x80483AB: main (example.c:11) 

這里第一行報告了內存"definitely lost",也就是說一定會存在內存泄漏,并給出了問題出現的位置。

實際上除了"definitely lost",valgrind還會給出"probably lost"的報告,這兩種報告的含義是這樣的:

  •  "definitely lost":你的程序一定存在內存泄漏問題,修復。
  •  "probably lost":你的程序看起來像是有內存泄漏,有可能你在使用指針完成一些特定操作,因此不一定100%存在問題。

總結

編寫正確的多線程代碼從來不是一件容易的事情,線程安全問題的根源在于共享資源,因此在使用共享資源前務必確認我們一定要用共享資源嗎? 

 

責任編輯:龐桂玉 來源: C語言與C++編程
相關推薦

2022-09-28 10:35:31

JavaScript代碼內存泄漏

2020-12-15 10:52:44

CIO企業網

2024-02-21 08:00:55

WindowsDWM進程

2025-05-06 15:31:17

陶哲軒AI工具

2022-07-08 09:43:24

攜程酒店數據接口服務平臺

2025-07-09 03:00:00

2010-04-02 10:29:02

CentOS安裝

2024-01-30 10:12:00

Java內存泄漏

2012-08-03 09:51:55

程序員編程

2012-07-04 14:40:37

Ajax

2017-01-05 19:34:06

漏洞nodejs代碼

2018-10-25 15:24:10

ThreadLocal內存泄漏Java

2024-03-22 13:31:00

線程策略線程池

2019-11-20 15:02:45

Java虛擬機內存

2016-11-24 15:54:06

androidJSONObject

2012-06-05 00:26:58

程序員

2019-05-09 14:42:41

安吉智能倉庫

2010-09-26 15:38:33

JVM內存泄漏

2023-03-17 07:44:24

IntelDDR4內存

2022-05-31 06:07:45

Excel表Python
點贊
收藏

51CTO技術棧公眾號

中文av一区特黄| 亚洲国产裸拍裸体视频在线观看乱了中文| 日本久久电影网| 亚洲春色在线| 亚洲第一天堂在线观看| 亚洲看片一区| 亚洲精品成a人ⅴ香蕉片| 精品一区二区在线看| 欧美乱大交xxxxx| 波多野结衣影院| 精品三级在线| 午夜精品免费在线观看| 亚洲欧洲精品一区二区| 欧美熟妇交换久久久久久分类| 日韩精品福利网| 欧美激情一区二区三区久久久 | 久久66热偷产精品| 国内伊人久久久久久网站视频 | 亚洲色图.com| 国产自摸综合网| 国产三级av片| 午夜视频精品| 最近中文字幕2019免费| 日韩无码精品一区二区| 成人国产精品久久| 色婷婷综合久色| 日本在线视频www色| 黄色毛片在线观看| 日韩在线亚洲| 91福利在线导航| 大伊香蕉精品视频在线| 久久精品视频免费看| 久久精品这里都是精品| 国产精品久久久久久免费观看 | 欧美日韩高清不卡| 37pao成人国产永久免费视频| 国产精品扒开做爽爽爽的视频| 久久蜜桃av一区二区天堂| 国产 高清 精品 在线 a| 国产又粗又猛又爽| 欧美成人精品欧美一级| 国产成人免费| 在线欧美日韩国产| 欧美韩国日本在线| 国产不卡123| 亚洲午夜电影网| 国产亚洲精品久久久久久久| 秋霞成人影院| 国产精品久久毛片av大全日韩| 免费一区二区三区| 亚洲日本中文字幕在线| av中文字幕在线不卡| 成人黄色在线免费观看| 性生交生活影碟片| 国产福利精品一区| 成人91视频| 高清毛片aaaaaaaaa片| 国产电影一区二区三区| 亚洲最大激情中文字幕| jlzzjlzzjlzz亚洲人| 国产麻豆精品久久一二三| 亚洲精品免费一区二区三区| 国产偷拍一区二区| 国产精品自拍在线| 91黄色国产视频| 亚洲卡一卡二卡三| 不卡一区二区中文字幕| 久久本道综合色狠狠五月| 日本大臀精品| 中文字幕免费不卡| 一区二区三区四区五区视频| 国产网站在线免费观看| 一区二区在线观看av| www.成年人视频| 悠悠资源网亚洲青| 91黄色免费版| www.51色.com| 77成人影视| 亚洲精品资源美女情侣酒店| 第一次破处视频| 香蕉久久网站| 狠狠久久亚洲欧美| 尤物精品国产第一福利三区| 波多野结衣欲乱| 欧美私人啪啪vps| 欧洲精品毛片网站| 一本色道久久综合无码人妻| 国产麻豆91精品| 久久本道综合色狠狠五月| 91这里只有精品| 亚洲一区二区综合| 国产成人精品无码播放| 亚洲三级久久久| 久久躁狠狠躁夜夜爽| 日韩三级在线观看视频| 欧美久久九九| 日韩av片永久免费网站| 97人妻一区二区精品免费视频 | 黄频网站在线观看| 久久久99精品久久| 亚洲天堂第一区| 日本不卡网站| 7777精品伊人久久久大香线蕉最新版| 亚洲少妇一区二区三区| 国产一区二区三区不卡视频网站 | 色a资源在线| 黑人巨大精品欧美一区二区一视频 | 超碰最新在线| 色噜噜狠狠成人网p站| wwwxxxx在线观看| 欧美日韩中文字幕一区二区三区| 色综合视频一区中文字幕| 五月婷婷激情视频| 粉嫩av一区二区三区粉嫩| 日本视频一区二区不卡| 久久国产精品黑丝| 欧美日韩一区二区在线观看| 韩国成人av| 国产成人美女视频| 91麻豆精品| 亚洲毛片一区二区| 久久久久久久久久久久久久免费看 | 久久wwww| 在线观看日韩www视频免费| 欧美日韩在线视频免费播放| 日韩av一区二| 久久精品ww人人做人人爽| 国产在线看片| 欧美制服丝袜第一页| 午夜av免费看| 欧美国产三级| 国产日本欧美一区| 高清毛片aaaaaaaaa片| 亚洲欧美欧美一区二区三区| 麻豆三级在线观看| 网红女主播少妇精品视频| 欧美成人一区在线| 一级成人免费视频| 欧美激情资源网| 久草精品在线播放| 天堂综合网久久| 久久久亚洲精品视频| a天堂视频在线| 亚洲男同性恋视频| 999在线精品视频| 91综合网人人| 91久久久久久久久久| 91精彩视频在线观看| 欧美性一二三区| 国产真人做爰视频免费| 三级欧美在线一区| 国产综合 伊人色| 黄在线观看免费网站ktv| 精品粉嫩超白一线天av| 久久精品国产亚洲AV无码男同| 国产不卡免费视频| av无码久久久久久不卡网站| av综合网页| 国内外成人免费激情在线视频网站| 黄色片一区二区三区| 午夜电影一区二区三区| 亚洲久久久久久| 亚洲一区二区三区免费在线观看| 久久99精品久久久久久青青日本| 末成年女av片一区二区下载| 国产视频久久久久| 色老头在线视频| 中文字幕va一区二区三区| 9l视频白拍9色9l视频| 久久免费大视频| 91精品在线国产| 日本孕妇大胆孕交无码| 亚洲国产高清高潮精品美女| 西西44rtwww国产精品| 久久婷婷久久一区二区三区| 性欧美videossex精品| 亚洲第一偷拍| 国产精品国模大尺度私拍| 美女av在线免费看| 亚洲性无码av在线| 国产精品久久免费| 亚洲国产视频直播| 能免费看av的网站| 国模少妇一区二区三区 | 99综合在线| 日本精品一区二区三区高清 久久| 97精品国产综合久久久动漫日韩 | 久久婷婷麻豆| 伊人久久99| 极品束缚调教一区二区网站| 国产精品福利在线观看网址| 成人日日夜夜| 亚洲久久久久久久久久| 国产精品天天操| 精品日韩美女的视频高清| 亚洲av毛片基地| 国v精品久久久网| 动漫av免费观看| 欧美精品首页| 日韩成人av电影在线| 久久久国产精品入口麻豆| 欧美夜福利tv在线| 粗大黑人巨茎大战欧美成人| 亚洲精品一区二区三区婷婷月| 91国产免费视频| 欧美视频在线观看免费| 男人的午夜天堂| 久久亚洲精品国产精品紫薇| 亚洲一区二区偷拍| 媚黑女一区二区| 视色,视色影院,视色影库,视色网| 久久不见久久见国语| 99久久精品免费看国产一区二区三区| 另类激情视频| 久久人人97超碰精品888| av电影在线观看| 亚洲免费一在线| 亚洲精品综合网| 91精品国产综合久久久久久| 销魂美女一区二区| 午夜精品爽啪视频| 欧美精品videos极品| 日本一区二区久久| 国产精品第七页| 高潮精品一区videoshd| 中文字幕丰满乱码| 日韩精品亚洲一区二区三区免费| 国产av熟女一区二区三区| 成人羞羞网站入口| 欧美日韩亚洲免费| 欧美aaaaa级| 成人一区二区三区四区| 不卡精品视频| 国产综合视频在线观看| 精品无人乱码一区二区三区| 国产91对白在线播放| 91桃色在线观看| 久久99青青精品免费观看| 日本精品在线| xvideos亚洲| 免费观看成人高潮| 丝袜情趣国产精品| 国产毛片在线看| 亚洲香蕉成视频在线观看 | 成人av资源在线观看| av在线免费观看不卡| 国产一区视频导航| 国产乱码一区二区三区四区| 久久国产精品区| 天堂一区在线观看| 麻豆精品一区二区| 色乱码一区二区三区在线| 美国三级日本三级久久99| 另类小说第一页| 奇米色一区二区| 向日葵污视频在线观看| 久久国产精品露脸对白| 亚洲精品www.| 激情六月婷婷久久| 999热精品视频| 国产酒店精品激情| 99日在线视频| 国产成人综合在线观看| 亚洲v在线观看| 91在线精品一区二区三区| 91精品小视频| 久久精品这里都是精品| 刘亦菲国产毛片bd| 亚洲日本中文字幕区| 麻豆一区产品精品蜜桃的特点| 亚洲午夜一区二区| 91av在线免费视频| 在线精品视频免费播放| 在线观看免费高清视频| 宅男在线国产精品| 欧美在线精品一区二区三区| 精品五月天久久| 在线播放毛片| 欧美激情视频一区二区三区不卡| 国产激情在线播放| 国产精品国模在线| www.成人在线.com| 国产伦精品一区二区三区在线| 久久99高清| 在线视频一二三区| 亚洲精品看片| 男女污污的视频| 国产精品99久久久久久久vr | 天堂va欧美ⅴa亚洲va一国产| 国产日韩精品推荐| 欧美色婷婷久久99精品红桃| 永久免费在线看片视频| 国产欧美日韩一级| 亚洲激情在线看| k8久久久一区二区三区| 日本人亚洲人jjzzjjz| 亚洲免费观看高清在线观看| 国产成人在线视频观看| 欧美午夜片在线看| 国模私拍视频在线| 中文字幕亚洲精品| 2018av在线| 国产精自产拍久久久久久| 加勒比久久高清| 综合一区中文字幕| 噜噜噜91成人网| 又黄又爽又色的视频| 国产欧美日韩综合精品一区二区| 国内偷拍精品视频| 欧美中文字幕一区二区三区| 后进极品白嫩翘臀在线视频| 色av中文字幕一区| 亚洲欧美韩国| 国产精品乱子乱xxxx| 97视频精品| 国产第一页视频| 成人免费视频app| av黄色免费在线观看| 日韩欧美在线网址| 黄色a在线观看| 久久影院免费观看| 免费视频观看成人| 久久精品日产第一区二区三区| 91精品国产91久久久久久密臀| 天天摸天天碰天天添| 成人午夜精品在线| avtt天堂在线| 欧美剧情电影在线观看完整版免费励志电影| 天堂av中文字幕| 欧美日韩国产999| 国产精品国产亚洲精品| 日产精品高清视频免费| 羞羞视频在线观看欧美| 亚洲精品第二页| 亚洲午夜日本在线观看| 国产情侣自拍小视频| 久久国内精品一国内精品| 欧美影视资讯| 欧美亚洲免费高清在线观看 | 91精品国产精品| caoporn成人免费视频在线| 日本久久高清视频| 久久国产精品第一页| 任你操精品视频| 欧美日韩亚洲综合| jizz在线免费观看| 国产精品激情自拍| 成人羞羞网站入口| 污污的网站免费| 1区2区3区国产精品| 91超薄丝袜肉丝一区二区| 深夜成人在线观看| 亚洲高清国产拍精品26u| 一本一道久久a久久综合精品| 秋霞av亚洲一区二区三| 人妻熟人中文字幕一区二区| 欧美中文字幕一二三区视频| 视频免费一区| 91亚洲午夜在线| 一区二区电影在线观看| 人妻换人妻仑乱| 性久久久久久久| 亚洲三级中文字幕| 国产精品电影网站| 欧美自拍偷拍| 在线视频观看一区二区| 亚洲欧美一区二区三区极速播放| 国产黄a三级三级三级| 欧美黑人xxxⅹ高潮交| 精品视频在线你懂得| 黄www在线观看| 中文字幕精品三区| 91麻豆成人精品国产| 久久99热精品| 爽爽窝窝午夜精品一区二区| 国内自拍视频一区| 日韩美女视频19| 好吊视频一二三区| 国产成人亚洲综合91精品| 欧美顶级大胆免费视频| 久久久久99人妻一区二区三区| 午夜久久久久久| 天堂а√在线资源在线| 亚洲自拍在线观看| 国产日韩欧美三级| 99久久99久久精品免费| 日韩视频免费观看高清完整版 | 高清福利在线观看| 91精品久久久久久久久中文字幕| 欧美日本精品| 男生草女生视频| 日韩亚洲电影在线| 亚洲天堂电影| 欧美日韩视频免费在线观看| 成人午夜大片免费观看| 中文在线免费看视频| 九九热视频这里只有精品| 国产精品密蕾丝视频下载 | 国产一区二区av在线| 国产99久久九九精品无码|