吴忠躺衫网络科技有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Windows CE中的內存泄漏問題

嵌入式應用開發 ? 來源:嵌入式應用開發 ? 作者:嵌入式應用開發 ? 2022-08-15 09:47 ? 次閱讀

很多實時嵌入式設備是長時間不間斷運行的,即使是少許的內存泄漏,也會積少成多,對嵌入式系統帶來災難性的影響。這幾天,我在嵌入式軟件項目中就飽嘗到這個痛苦,讓我明白到嵌入式實時系統的應用軟件也會有許多內存問題,從而導致嵌入式系統的崩潰。例如非法的內存訪問、各種死鎖以及諸如堆棧溢出、數組越界和內存泄漏等

Windows CE作為最流行的一種嵌入式操作系統,現正廣泛被應用。我所負責的嵌入式應用程序也是在Windows CE平臺上開發的。在進入測試階段中,我發現有一個程序模塊系統內存和CPU資源消耗急劇增加,持續增長到出現OutOfMemoryError為止,然后自動重啟。這個問題折騰到我生不如死,痛苦不堪?;宋液脦讉€通宵達旦的加班后,經過分析終于確認Windows CE內存泄漏是造成這次Windows CE系統崩潰的主要原因。這里與大家分享我在開發過程中遇到的內存泄漏的檢測和處理解決過程。

一.Windows CE如何進行內存分配?

為了判斷是否有內存泄露,我們首先需要了解Windows CE是如何管理內存的。許多嵌入式程序員都有一個共識,就是如果評選在Windows CE 程序中遇到最多的問題,那其中一個問題一定有內存問題。

(1)什么是Windows CE內存管理

一般來說,運行Windows CE的嵌入式設備出于緊湊型的考慮內存都不大,以至于有時候有些程序員會為了節省內存開支而犧牲程序的某些性能。但盡管WinCE系統的內存很小,用來管理內存的函數卻十分完善。Windows CE實現了Windows XP中幾乎全部的Win32內存管理API。例如,Windows CE支持虛擬內存分配,本地和分離的堆管理,甚至還有內存映射文件。像Windows XP一樣,Windows CE支持帶有應用程序間內存保護功能的32位地址空間,這一點對于多程序和多線程運行時是非常重要的功能。但是Windows CE畢竟是被設計來應用于實時場合的,所以它底層的內存結構又不同于Windows XP。

Windows CE內核可以在Flash上直接運行,也可以加載到內存中運行。Flash的運行方式,是把內核的可執行映像燒寫到Flash上,系統啟動時從Flash的某個地址開始執行。在這種情況下,Windows CE系統就像直接讀硬盤,存儲在Flash上的程序能夠以現場執行的方式運行。這種能力對小型系統來說使之在具有巨大的優勢,這樣這能快速啟動一個應用程序,因此這種方法被很多嵌入式系統所采用。另一種是內核加載方式,是把內核的壓縮文件存放在Flash上,系統啟動時讀取壓縮文件在內存里解壓,然后開始執行。

(2)虛擬內存和函數應用

和大多數現代操作系統一樣,Windows CE實現按需調頁的虛擬內存機制。由于Windows CE系統使用了虛擬內存,這就給應用程序造成了一個假象,以為計算機安裝的內存遠遠超過自己所需要的數量。Windows CE是32位的操作系統,因此支持4GB的虛擬地址空間。Windows把這些地址空間分給進程和系統使用,每個部分可以獲得2GB的虛擬內存。

虛擬內存是內存類型中最基礎的。Windows CE 實現了系統的虛擬內存管理,在一個虛擬內存系統中,應用程序主要處理這個虛擬的地址空間,并不涉及到由硬件管理的物理內存。系統調用虛擬內存API來為其它類型內存分配內存,包括堆和棧。Windows CE虛擬內存頁可以處在三種狀態:自由(free),保留(reserved),或被提交(committed)。

簡單說,就是當一個應用程序要查詢系統的內存時,可使用虛擬內存API,包括VirtualAlloc,VirtualFree和VirtualReSize函數,這些函數可以直接操作虛擬內存空間的虛擬內存頁面。例如,頁面可以保留,提交給物理內存,或使用這些函數釋放。Windows CE實現了Win32的GetSystemInfo和GlobalMemoryStatus函數。另一個檢測系統狀態的函數是:void GlobalMemoryStatus(LPMEMORYSTATUS lpmst),通過GlobalMemoryStatus返回的信息可以驗證Windows CE內存結構。

(3)釋放虛擬內存

不同于Windows XP,Windows CE只支持在堆中分配固定(fixed)的塊。這簡化了內存塊在堆中的處理,但是這使得堆在分配和釋放一段時間后會產生碎片。當堆里已經清空的時候,仍然會占用大量的虛擬內存頁,因為系統不能在堆中內存頁沒有完全釋放的時候回收這些頁。這時,一般情況下是可以通過調用VirtualFree來取消提交,或釋放虛擬內存。從物理RAM頁中取消提交或者取消映射,但是保持頁被保留的狀態,當在區域中的所有的頁通過VirtualFree被釋放時,也應該處在同樣的情況下。更確切地說,區域中的全部頁要被釋放,那這些頁要么都是被提交的頁,要么都是被保留的頁。如果有些頁被提交,有些頁被保留,那么VirtualFree函數調用就會失敗。

實際上,Windows CE會監視系統自由的內存,并對越來越少的內存作出響應。當很少內存可用時,Windows CE首先發送WM_HIBERNATE消息,接下來會限制可能的內存分配。當應用程序被發送了一個WM_HIBERNATE消息后,系統將檢測內存級別,確認是否可用內存在限度之上,如果可用內存不足,WM_HIBERNATE消息將被發送給下一個程序,這會持續到所有程序被發送了WM_HIBERNATE消息。

二. 什么是內存泄露

Win32編程中常見內存錯誤:①內存分配錯誤;②使用未初始化的內存;③內存泄露;④使用已經釋放的內存資源。

poYBAGL5pPWACkyiAADQEXbnnQk566.png

(1)什么是內存泄漏

內存泄漏是指程序在運行過程中申請的內存,在程序結束時沒有被釋放。我們常說的內存泄漏是指堆內存的泄漏,堆內存是指程序從堆中分配的。一般來說,應用程序是使用從堆中分配到一塊內存,使用完后程序必須負責相應的釋放該內存塊。否則,這塊內存就不能被再次使用,我們就說這塊內存泄漏了。

一般來說,在所有時刻Windows CE內存管理器都知道進程所擁有的物理內存和虛擬內存。然而,如果進程分配內存時但由于Bug而無法釋放內存(內存泄漏),內存管理器就可能無法了解這些已分配的內存,也無法重新訪問這些內存,而必須等到進程退出時回收內存。但需要特別注意的是,同樣的程序在Window XP平臺上可能沒有什么問題,但在缺乏內存的Windows CE平臺,經過長時間運行該程序可能會內存耗盡而導致系統重啟,這是我在經過幾個生不如死的通宵達旦測試后得到的寶貴經驗和教訓。

因此,內存泄漏引發的性能失常完全不同于程序錯誤,這些問題很難通過調試器對代碼進行單步調試加以解決。對于將會在某時刻退出的桌面應用程序,較小的內存泄漏是可以承受的,因為退出進程將把占用的所有內存返還給操作系統。但對于長時間運行的嵌入式系統,則通常需要確保絕對沒有內存泄漏。

(2)常見的內存泄漏原因

常見的內存泄漏有這幾種原因:①Windows CE內存碎片。②在局部堆申請的堆只增加不會馬上減少,直到程序退出。③程序運行時分配物理內存,當程序使用完后,這些物理內存仍然被占用,直到系統內存不足時分頁內存交換到分頁文件中,然后才釋放掉其占用的物理內存。④Windows CE內存管理的缺陷。

總而言之,內存泄漏產生的主要原因是保留了卻不再使用的內存空間。Windows CE雖然有自動管理內存的功能,但內存泄漏也是不容忽視,它往往是破壞嵌入式系統穩定性的重要因素。

三. 如何檢測和處理內存泄漏?

如何查找引起內存泄漏的原因,一般有兩個步驟:第一是安排有經驗的編程人員對代碼進行走查和分析,找出內存泄漏發生的位置。第二是使用專門的內存泄漏測試工具進行測試。

(1)代碼走讀檢測內存泄漏

通常在懷疑發生內存泄漏之后,第一步是要弄清楚泄漏了什么數據和引起了什么泄漏。一般說來,一個正常的系統在其運行穩定后其內存的占用量是基本穩定的,不應該是無限制的增長的。根據這樣的基本假設,我們持續地觀察系統運行時使用的內存的大小,如果內存的大小持續地增長,則說明系統存在內存泄漏。

內存泄漏可通過代碼走讀來發現和定位,也可以用專用的工具來測試和定位。實際上,對于內存泄漏,代碼檢查有時能比采用任何技術解決方案更快地找到問題所在。預防內存泄漏的唯一方法就是要求程序員在程序結束時,必須將每個申請的內存都釋放。

(2)使用工具檢測內存泄漏

一旦知道確實發生了內存泄漏,就需要更專業的工具來查明為什么會發生泄漏。在這個時候,我們通常需要使用一些開銷較低的工具來監控和查找內存泄漏。查找內存泄漏的工具很多,最常用的釋放工具就是dmalloc和mpatrol,這些工具提供了記錄并檢查所有內存分配的調試版堆棧,從而有利于分析內存泄漏和懸掛指針。

檢測內存泄漏的關鍵是要能截獲住對分配內存和釋放內存的函數的調用,當截獲住這兩個函數,我們就能跟蹤每一塊內存的生命周期。比如,每當成功的分配一塊內存后,就把它的指針加入一個全局的list中;每當釋放一塊內存,再把它的指針從list中刪除。這樣,當程序結束的時候,list中剩余的指針就是指向那些沒有被釋放的內存。哪么,最簡單的內存泄漏檢測方式就是截獲住這些指針。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 嵌入式
    +關注

    關注

    5092

    文章

    19177

    瀏覽量

    307664
  • WINDOWS
    +關注

    關注

    4

    文章

    3569

    瀏覽量

    89303
  • 內存泄漏
    +關注

    關注

    0

    文章

    39

    瀏覽量

    9235
收藏 人收藏

    評論

    相關推薦

    Windows CEWindows 2000/XP有什么區別?

    Windows系統。它使用平面內存模式尋址,可以同時運行多個程序并支持一個程序的多個線程,并且非常精煉,只有很小的內存要求。
    發表于 10-15 08:10

    Windows CE板卡支持包介紹

    Windows CE板卡支持包用于netX的Windows CE 鏡像帶有一個周期為120天的免費測試版。板卡支持包需要通過專用的驅動創建自己的鏡像。其中包括
    發表于 04-11 08:50 ?27次下載

    基于Windows CE嵌入式系統的智能小區廣告機的設計

    Windows CE 是微軟公司專門為嵌入式市場設計,為快速建立下一代小內存占用的設備提供的一個健壯的實時操作系統。Windows CE
    發表于 08-18 10:15 ?19次下載

    ARM嵌入式WINDOWS CE實踐教程

    ARM嵌入式WINDOWS CE實踐教程 安裝WINDOWS CE 4.2本節講述如何安裝Windows
    發表于 03-01 15:45 ?81次下載

    Windows CE .NET 常見問題與解答

    Windows CE .NET 常見問題與解答 1. Microsoft Windows ® CE .NET®是什么?Microsoft
    發表于 07-07 10:09 ?1565次閱讀
    <b class='flag-5'>Windows</b> <b class='flag-5'>CE</b> .NET 常見問題與解答

    Windows CE簡介、特點及應用

    Windows CE簡介、特點及應用 一、Windows CE簡介  
    發表于 01-11 08:43 ?4200次閱讀

    Windows CE詳解

    從純粹軟件技術的角度看 Windows CE ,則可以說Windows CE就是嵌入式的Windows,雖然
    發表于 06-09 15:51 ?67次下載

    Windows CE下實現串口通信

    文章首先簡單介紹了在WIndows CE下設備驅動程序的訪問 方式, 然后詳細介紹了在WIndows CE下實現串口通信的方法和過程。
    發表于 04-18 14:12 ?6次下載

    Windows CE 進程、線程和內存管理三

    三、內存管理 同其它Windows操作系統一樣,Windows CE.NET也支持32位虛擬內存機制、按需分配
    發表于 11-08 10:30 ?0次下載
    <b class='flag-5'>Windows</b> <b class='flag-5'>CE</b> 進程、線程和<b class='flag-5'>內存</b>管理三

    內存泄漏的特點和類型

    在計算機科學,內存泄漏(memory leak)指由于疏忽或錯誤使程序未能釋放而造成不能再使用的內存的情況。內存
    的頭像 發表于 06-20 10:58 ?2867次閱讀

    什么是內存泄漏?內存泄漏有哪些現象

    內存泄漏幾乎是很難避免的,不管是老手還是新手,都存在這個問題,甚至 Windows 與 Linux 這類系統軟件也或多或少存在著內存泄漏。
    的頭像 發表于 09-05 17:24 ?9790次閱讀

    怎么解決C語言中的內存泄漏問題呢?

    只有在堆內存里面才會發生內存泄漏的問題,在棧內存不會發生內存
    發表于 06-11 17:31 ?588次閱讀
    怎么解決C語言中的<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>問題呢?

    什么是內存泄漏?如何避免JavaScript內存泄漏

    JavaScript 代碼中常見的內存泄漏的常見來源: 研究內存泄漏問題就相當于尋找符合垃圾回收機制的編程方式,有效避免對象引用的問題。
    發表于 10-27 11:30 ?450次閱讀
    什么是<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>?如何避免JavaScript<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>

    線程內存泄漏問題的定位

    記錄一個關于線程內存泄漏問題的定位過程,以及過程的收獲。 1. 初步定位 是否存在內存泄漏:想到內存
    的頭像 發表于 11-13 11:38 ?654次閱讀
    線程<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>問題的定位

    C語言內存泄漏問題原理

    內存泄漏問題只有在使用堆內存的時候才會出現,棧內存不存在內存泄漏問題,因為棧
    發表于 03-19 11:38 ?574次閱讀
    C語言<b class='flag-5'>內存</b><b class='flag-5'>泄漏</b>問題原理
    百家乐官网注码投注论坛| 大发888备用地址| 巴特百家乐的玩法技巧和规则 | 百家乐是骗人的| 金沙百家乐现金网| 澳门百家乐经历| 百家乐稳赢投资法| 百家乐平注法到6568| 太阳城83| 德州扑克书| bet365体育| 百家乐官网试用软件| 西青区| 大发888我发财官网| 赌球网址| 盛世国际投注| 百家乐官网声音不印网| 百家乐官网买闲打法| 永利百家乐官网的玩法技巧和规则 | 百家乐官网翻天百度影音| 开心8百家乐游戏| 如何看百家乐路| 大发888casino下载| 鸿发| 博彩百家乐官网在线| 鸟巢百家乐官网的玩法技巧和规则| 皇马百家乐官网的玩法技巧和规则| 使用的百家乐软件| 大发888娱乐城打发888打发8| 皇冠在线代理| 百家乐官网六手变化混合赢家打法 | 博彩通天上人间| 赌博百家乐官网的乐趣| 风水24山辛山乙| 最佳场百家乐的玩法技巧和规则| 大发888怎么刷钱| 博御国际| 闲和庄百家乐官网娱乐平台| 金百亿百家乐娱乐城| 大发888娱乐场168| 在线百家乐官网有些一|