• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆-379  評論-37  文章-0  trackbacks-0

            轉:http://www.cnblogs.com/minggoddess/archive/2010/12/15/1907179.html

            動態鏈接庫中分配內存引起的

                   本文主要是探討關于在動態鏈接庫分配的內存在主程序中釋放所產生的問題,該問題是我在剛做的PJP工程中所遇到的,由于剛碰到之時感動比較詭異(這也是學識不夠所致),所以將它寫下來,大家一起分享.

                   問題來由:

            由于該工程中要用到聲音,所以我的分工之一就是用DirectMusic和DirectSound來開發聲音播放的動態庫,以提供給該工程的兩個部分:仿真控制部分( 語音 )和三維部分( 場景聲音 )使用,兩個工程中的聲音都以單獨的線程播放,且兩個線程幾乎相同.,然而該動態庫在以前的運行中一直沒有出現過問題, 直到工程開發即將要結束階段的前一個星期,我碰才到了這個問題,首先是三維部分中聲音在第一次播放是沒有問題,在播放第二個聲音就出了問題(老是這樣),但是仿真控制部分還是沒有出現問題,我查不出錯誤,就用前一天的三維的版本來運行,重新生成后聲音播放沒有問題,初步以為是我同事當天寫的代碼有問題,可是第二天,同樣的問題又發生了,還是那樣,三維有問題,仿真部分沒問題,怎么回事呢?我將三維的聲音線程寫成和仿真部分一樣的,還是三維出問題,仿真沒問題.我不相信是動態庫中出問題,同一個動態庫,同樣的代碼,只是在兩個不同的程序里,為什么一個出問題,而另一個不出問題呢?

                   出錯信息如下:

                              windows已在PowipD.exe中觸發一個斷點.
                              其原因可能是堆被損壞,這也說明PowipD.exe中或它所加載的任何DLL中有bug.
                              輸出窗口可能提供了更多診斷信息

                   按下中斷后輸入窗口出現的信息如下:

                           HEAP[PowipD.exe]:Invalid Address specified to RtlValidateHeap( 01CC0000, 03723758 )
                           Windows已在PowipD.exe中觸發一個斷點.
                           輸出窗口可能提供了更多診斷信息

                   由于這些原因讓我迷惑不解,到底是為什么出現這種奇怪的事情呢?問題出在那里?

             

            問題的解決:

            簡單地說:DLL中分配的內存DLL要負責釋放!(一個模塊分配的內存要在同一個模塊中釋放!)

            找到解決方法后我查看了三個工程的設置(三個工程皆用VS2005開發):

                   三維:使用MDd(多線程調試DLL)運行期庫

                   仿真控制:使用MTd(多線程調試)運行期庫

                   聲音動態庫:使用MDd(多線程調試DLL)運行期庫

            于是,我將三個工程皆改為:使用MDd(多線程調試DLL)運行期庫時,問題解決。

            但是如果三個工程中有的不是用C/C++寫的,或者其中有工程的設置已經不便更改了,那又有什么辦法呢?

            該問題主要是關于DLL與進程的地址空間的問題(下面是<<核心編程>>中的一段話):

            單個地址空間是由一個可執行模塊和若干個DLL模塊組成,這些模塊中,有些可以鏈接到靜態版本的C/C++運行期庫, 有些可以鏈接到一個DLL版本的運行期庫,而有些模塊(如果不是用C/C++編寫的話)則根本不需要C/C++運行期庫,許多開發人員經常會犯一個常見的錯誤,因為他們忘記了若干個C/C++運行期庫可以存在于單個地址空間中.請看下面的代碼(下面代碼不是書上的):

            DLL中如下:

                   int *DllFunc()

            {

                   int *p = new int;

                   return p;

            }

             

            EXE中如下:

            void EXEFunc()

            {

                   int *p = DllFunc();

                   if( p!= NULL )

                          delete p;

            }

             

            如何看待這個問題呢?上面的代碼能夠正確運行嗎?DLL函數分配的內存是由EXE的函數釋放的嗎?答案是可能的.上面顯示的代碼并沒有提供足夠的信息.如果EXE和DLL都鏈接到DLL的C/C++運行期庫,那么上面的代碼將能夠很好地運行.但是,如果兩個模塊中的一個或者兩個鏈接到靜態C/C++運行期庫,那個對delete的操作就會失敗.

            一個很方便的方法可以解決這個問題.當一個模塊提供一個用于分配內存塊的函數時,該模塊也必須提供相應的釋放內存的函數,將上面的代碼改成如下的樣子就不會出錯了:

             

            DLL中如下:

                   int *DllFunc()

            {

                   int *p = new int;

                   return p;

            }

             

            void DLLDelete( int *p)

            {

                   if( p != NULL )

                          delete p;

            }

             

            EXE中如下:

            void EXEFunc()

            {

                   int *p = DllFunc();

                   DLLDelete( p );

            }

            這樣的代碼才是正確的,當我將代碼改成如上類似后,再將三維的運行期庫改為原來一樣(使用MDd(多線程調試DLL)運行期庫),三維的聲音也正確無誤地播放出來了。當你在編寫一個模塊時,你時刻都得記著,別人的代碼不一定是用C/C++寫的,也有可能不能同時鏈接到DLL的C/C++運行期庫。malloc和free函數也有類似的問題。

            當然在你在生成你的模塊時是可以選擇運行期庫的,VC 6.0配有6個運行期庫:其描述如下:

             

            庫名

            描述

            LibC.lib

            用于單線程應用程序的靜態應用程序的靜態鏈接庫      

            LibCD.lib

            用于單線程應用程序的靜態鏈接庫的調試版

            LibCMt.lib

            用于多線程應用程序的靜態鏈接庫的發行版

            LibCMtD.lib

            用于多線程應用程序的靜態鏈接庫的調試版

            MSVCRt.lib

            用于動態鏈接MSVCRt.dll庫的發行版的輸入庫

            MSVCRtD.lib

            用于動態鏈接MSVCRtD.dll的調試版的輸入庫。該庫同時支持單線程應用程序和多線程應用程序

             

            而VS2005中只配了4個C/C++運行期庫,就是上表中的后面4個。

            建議各位在軟件開發時同一軟件的不同模塊最好使用一致的運行庫,否則還可能出現鏈接問題。

            posted on 2012-06-02 17:52 小王 閱讀(1786) 評論(1)  編輯 收藏 引用 所屬分類: VC

            評論:
            # re: 動態鏈接庫分配的內存在主程序中釋放所產生的問題 2012-06-03 02:03 | egmkang
            程序的接口有問題.
            誰申請的,誰釋放;誰提供申請的接口,誰也提供釋放的接口  回復  更多評論
              
            精品久久久久久亚洲精品 | 99久久精品免费看国产免费| 国产精品久久精品| 久久综合精品国产一区二区三区| 久久青青草视频| 国内精品久久人妻互换| 久久久久99精品成人片牛牛影视| 久久天天躁狠狠躁夜夜躁2014| 久久99国产精品久久99果冻传媒 | 久久久久99精品成人片| 狠狠综合久久AV一区二区三区| 久久99精品国产自在现线小黄鸭 | 99国内精品久久久久久久| 国产免费久久精品99re丫y| 久久国产精品99精品国产| 久久强奷乱码老熟女| 狠狠色丁香婷综合久久| 久久AV高潮AV无码AV| 深夜久久AAAAA级毛片免费看| 久久精品国产亚洲沈樵| 久久久久女人精品毛片| 色天使久久综合网天天| 久久久久亚洲爆乳少妇无 | 狠狠综合久久AV一区二区三区| 国产精品gz久久久| 精品久久久久久国产潘金莲| 久久香综合精品久久伊人| 色婷婷噜噜久久国产精品12p| 99精品伊人久久久大香线蕉| 久久青草国产精品一区| 久久夜色精品国产噜噜噜亚洲AV| 一97日本道伊人久久综合影院| 亚洲精品国产成人99久久| 久久精品国产亚洲AV嫖农村妇女| 亚洲精品无码久久不卡| 久久久久国产视频电影| 久久精品亚洲福利| 久久久久久亚洲精品无码| 久久av免费天堂小草播放| 精品无码久久久久久久久久| 国产农村妇女毛片精品久久 |