• <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要負責釋放!(一個模塊分配的內存要在同一個模塊中釋放?。?/p>

            找到解決方法后我查看了三個工程的設置(三個工程皆用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 小王 閱讀(1783) 評論(1)  編輯 收藏 引用 所屬分類: VC

            評論:
            # re: 動態鏈接庫分配的內存在主程序中釋放所產生的問題 2012-06-03 02:03 | egmkang
            程序的接口有問題.
            誰申請的,誰釋放;誰提供申請的接口,誰也提供釋放的接口  回復  更多評論
              
            国产麻豆精品久久一二三| 精品人妻久久久久久888| 精品久久久久久中文字幕| 午夜久久久久久禁播电影| 亚洲午夜久久久久妓女影院| 亚洲国产成人久久综合碰| 人妻无码久久精品| 99久久婷婷国产综合亚洲| 国产精品久久久天天影视| 精品久久久久久久久久久久久久久| 国产无套内射久久久国产| 中文字幕无码免费久久| 国产欧美一区二区久久| 亚洲中文字幕久久精品无码APP | AV无码久久久久不卡蜜桃| 好久久免费视频高清| 日日躁夜夜躁狠狠久久AV| 久久久国产精华液| 国产精品99久久久久久宅男| 国内精品久久久久久99蜜桃| 四虎影视久久久免费观看| 久久综合久久综合久久| 久久99精品国产麻豆宅宅| 欧美喷潮久久久XXXXx| 中文字幕精品久久久久人妻| 国产亚洲精午夜久久久久久| 久久精品国产99国产精品澳门| 久久久久人妻一区精品色| 久久亚洲精精品中文字幕| 久久久久久夜精品精品免费啦| 热re99久久精品国99热| 久久夜色精品国产欧美乱| 久久夜色精品国产噜噜麻豆| 久久国产一区二区| 久久久噜噜噜久久| 人妻久久久一区二区三区| 嫩草影院久久99| 中文字幕无码精品亚洲资源网久久| 国产综合久久久久| 久久夜色精品国产噜噜亚洲a| 成人国内精品久久久久影院|