青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

DLL中傳遞STL參數,vector對象作為dll參數傳遞等問題(轉)

 


STL跨平臺調用會出現很多異常,你可以試試.

STL使用模板生成,當我們使用模板的時候,每一個EXE,和DLL都在編譯器產生了自己的代碼,導致模板所使用的靜態成員不同步,所以出現數據傳遞的各種問題,下面是詳細解釋。

原因分析:
一句話-----如果任何STL類使用了靜態變量(無論是直接還是間接使用),那么就不要再寫出跨執行單元訪問它的代碼。 除非你能夠確定兩個動態庫使用的都是同樣的STL實現,比如都使用VC同一版本的STL,編譯選項也一樣。強烈建議,不要在動態庫接口中傳遞STL容器!!

STL不一定不能在DLL間傳遞,但你必須徹底搞懂它的內部實現,并懂得為何會出問題。
微軟的解釋:
http://support.microsoft.com/default.aspx?scid=kb%3ben-us%3b172396
微軟給的解決辦法:
http://support.microsoft.com/default.aspx?scid=kb%3ben-us%3b168958

1、微軟的解釋:
大部分C++標準庫里提供的類直接或間接地使用了靜態變量。由于這些類是通過模板擴展而來的,因此每個可執行映像(通常是.dll或.exe文件)就會存在一份只屬于自己的、給定類的靜態數據成員。當一個需要訪問這些靜態成員的類方法執行時,它使用的是“這個方法的代碼當前所在的那份可執行映像”里的靜態成員變量。由于兩份可執行映像各自的靜態數據成員并未同步,這個行為就可能導致訪問違例,或者數據看起來似乎丟失或被破壞了。

可能不太好懂,我舉個例子:假如類A<T>有個靜態變量m_s,那么當1.exe使用了2.dll中提供的某個A<int>對象時,由于模板擴展機制,1.exe和2.dll中會分別存在自己的一份類靜態變量A<int>.m_s。
這樣,假如1.exe中從2.dll中取得了一個的類A<int>的實例對象a,那么當在1.exe中直接訪問a.m_s時,其實訪問的是 1.exe中的對應拷貝(正確情況應該是訪問了2.dll中的a.m_s)。這樣就可能導致非法訪問、應當改變的數據沒有改變、不應改變的數據被錯誤地更改等異常情形。

原文:
Most classes in the Standard C++ Libraries use static data members directly or indirectly. Since these classes are generated through template instantiation, each executable image (usually with DLL or EXE file name extensions) will contain its own copy of the static data member for a given class. When a method of the class that requires the static data member is executed, it uses the static data member in the executable image in which the method code resides. Since the static data members in the executable images are not in sync, this action could result in an access violation or data may appear to be lost or corrupted.

1、保證資源的分配/刪除操作對等并處于同一個執行單元;
   比如,可以把這些操作(包括構造/析構函數、某些容器自動擴容{這個需要特別注意}時的內存再分配等)隱藏到接口函數里面。換句話說:盡量不要直接從dll中輸出stl對象;如果一定要輸出,給它加上一層包裝,然后輸出這個包裝接口而不是原始接口。

2、保證所有的執行單元使用同樣版本的STL運行庫。
   比如,全部使用release庫或debug庫,否則兩個執行單元擴展出來的STL類的內存布局就可能會不一樣。

只要記住關鍵就是:如果任何STL類使用了靜態變量(無論是直接還是間接使用),那么就不要再寫出跨執行單元訪問它的代碼。

解決方法:
1. 一個可以考慮的方案
比如有兩個動態庫L1和L2,L2需要修改L1中的一個map,那么我在L1中設置如下接口
int modify_map(int key, int new_value);
如果需要指定“某一個map”,則可以考慮實現一種類似于句柄的方式,比如可以傳遞一個DWORD
不過這個DWORD放的是一個地址

那么modify_map就可以這樣實現:
int modify_map(DWORD map_handle, int key, int new_value)
{
    std::map<int, int>& themap = *(std::map<int, int>*)map_handle;
    themap[key] = new_value;
}

map_handle的值也首先由L1“告訴”L2:
DWORD get_map_handle();

L2可以這樣調用:
DWORD h = get_map_handle();
modify_map(h, 1, 2);

2. 加入一個額外的層,就可以解決問題。所以,你需要將你的Map包裝在dll內部,而不是讓它出現在接口當中。動態庫的接口越簡單越好,不好去傳太過復雜的東東是至理名言:)

 

在動態連接庫開發中要特別注意內存的分配與釋放問題,稍不注意,極可能造成內存泄漏,從而訪問出錯。例如在某DLL中存在這樣一段代碼:

extent "C" __declspec(dllexport) 
void ExtractFileName( const std::string& path //!< Input path and filename.
, std::string& fname //!< Extracted filename with extension.
)
{
std::string::size_type startPos = path.find_last_of('\\');
fname.assign(path.begin() startPos 1, path.end() );
}

在DLL中使用STL對象std::string,并且在其中改變std::string的內容,即發生了內存的重分配問題,若在EXE中調用該函數會出現內存訪問問題。主要是:因為DLL和EXE的內存分配方式不同,DLL中的分配的內存不能在EXE中正確釋放掉。

解決這一問題的途徑如下:
一般情況下:構建DLL必須遵循誰分配就由誰釋放的原則,例如COM的解決方案(利用引用計數),對象的創建(QueryInterface)與釋放均在COM組件內部完成。在純C 環境下,可以很容易的實現類似方案。


在應用STL的情況下,很難使用上述方案來解決,因此必須另辟蹊徑,途徑有二:
1、自己寫內存分配器替代STL中的默認分配器。
2、使用STLport替代系統的標準庫。

其實,上述問題在VC7及以后版本中,已得到解決,注意DLL工程和調用的工程一定要使用多線程DLL庫,就不會發生內存訪問問題。

 

 

一個很奇怪的問題:DLL中使用std::string作為參數結果出錯

這段時間,在工程中將一些功能封裝成動態庫,需要使用動態庫接口的時候.使用了STL的一些類型作為參數.

比方string,vector,list.但是在使用接口的時候.
  1. class exportClass
  2. {
  3.      bool dll_funcation(string &str);
  4. };
復制代碼
//上面這個類只是一個形式,具體內容不寫出來了.這個類被導出

當我在使用這個庫的時候.這樣寫代碼:
  1. string str="":
  2. exportClass tmp;
  3. tmp.dll_function(str);
復制代碼
這個函數能成功調用.但是在函數里面會給這個數組附值.如果字符串太長,就會出錯.函數調用能成功,但是一旦str資源需要釋放的時候,資源就不能釋放了,提示釋放了錯誤的內存空間.

一點一點取掉這個函數的代碼.最后就剩下

str="qadasdasdasdsafsafas";

還是出錯誤.

如果改成很短的字符串,就不會出錯誤.
在這個時候,只能嘗試認為是字符串的空間太小

最終我修改成這樣,錯誤消失了.希望錯誤真的是這個引起的
  1. string str="":

  2. str.resize(1000);

  3. exportClass tmp;

  4. tmp.dll_function(str);

 

今天寫程序的時候要給一個模塊的dll傳遞一個參數,由于參數數量是可變的,因此設計成了vector<string>類型,但調試過程中發現在exe中的參數傳遞到dll中的函數后,vector變成空的,改成傳引用類型后,vector竟然變得很大,并且是無意義的參數。

對于這個問題,兩種辦法:

1.傳遞vector指針

2.傳遞const vector<TYPE>。

究其原因:

是因為vector在exe和dll之間傳遞的時候,由于在dll內可能對vector插入數據,而這段內存是在dll里面分配的,exe無法知道如何釋放內存,從而導致問題。而改成const類型后,編譯器便知道dll里不會改變vector,從而不會出錯。

或者可以說這是"cross-DLL problem."(This problem crops up when an object is created using new in one dynamically linked library (DLL) but is deleted in a different DLL)的一種吧。

對于STL,在DLL中使用的時候,往往存在這些問題,在網絡上搜集了下,這些都是要平時使用STL的時候注意的。

***************************************************************************************************************

引用http://www.hellocpp.net/Articles/Article/714.aspx

當template 遭遇到dynamic link 時候, 很多時候卻是一場惡夢.
現在來說說一部分我已經碰到過的問題. 問題主要集中在內存分配上.
1> 
      拿STL來說, 自己寫模板的時候,很難免就用到stl. stl的代碼都在頭文件里. 那么表示著內存分配的代碼.只有包含了它的cpp 編譯的時候才會被決定是使用什么樣的內存分配代碼. 考慮一下: 當你聲明了一個vector<> . 并把這個vector<>交給一個 dll里的代碼來用. 用完后, 在你的程序里被釋放了.    那么如果你 在dll里往vector里insert了一些東西. 那么這個時候insert 發生的內存分配的代碼是屬于dll的. 你不知道這個dll的內存分配是什么. 是分配在哪里的. 而這個時候.釋放那促的動作卻不在dll里.....同時. 你甚至無法保證編譯dll的那個家伙使用的stl版本和你是完全一樣的..>
      如此說來, 程序crash掉是天經地義的.... 
      對策: 千萬別別把你的stl 容器,模板容器在 dll 間傳來傳去 . 記住string也是....

2> 
     你在dll的某個類里聲明了一個vector之類的容器. 而沒有顯式的寫這個類的構造和析構函數. 那么問題又來了.
     你這個類肯定有操作這vector的函數. 那么這些函數會讓vecoter<>生成代碼. 這些代碼在這個dll里都是一致的. 但是別忘了.你沒有寫析構函數...... 如果這個時候, 別人在外面聲明了一個這樣的類.然后調用這個類的函數操作了這個vector( 當然使用者并不知道什么時候操作了vector) . 它用完了這個類以后. 類被釋放掉了. 編譯器很負責的為它生成了一份析構函數的代碼...... 聽好了.這份代碼并不是在 dll里 ... . 事情于是又和1>里的一樣了.... crash ......(可能還會伴隨著迷茫.....)
     對策: 記得dll里每個類,哪怕式構造析構函數式空的. 也要寫到cpp里去. 什么都不寫也式很糟糕的.....同時,更要把任何和內存操作有關的函數寫到 .cpp 里...

3> 
    以上兩個問題似乎都是比較容易的-----只要把代碼都寫到cpp里去, 不要用stl容器傳來傳去就可以了.
   那么第三個問題就要麻煩的多.
   如果你自己寫了一個模板, 這個模板用了stl 容器..........
   這個時候你該怎么辦呢?
 顯然你無法把和內存分配相關的函數都寫到.cpp里去 . template的代碼都必須放到header file里.....
   對策: 解決這個問題的基本做法是做一個stl 內存分配器 , 強制把這個模板里和內存分配相關的放到一個.cpp里去.這個時候編譯這個cpp就會把內存分配代碼固定在一個地方: 要么是dll. 要么是exe里...

模板+動態鏈接庫的使用問題還很多. 要千萬留心這個陷阱遍地的東西啊

***************************************************************************************************************************

微軟關于這類問題的解釋:

You may experience an access violation when you access an STL object through a pointer or reference in a different DLL or EXE

http://support.microsoft.com/default.aspx?scid=KB;en-us;q172396

How to export an instantiation of a Standard Template Library (STL) class and a class that contains a data member that is an STL object

http://support.microsoft.com/default.aspx?scid=KB;en-us;q168958

 

 

 

總結:

字符串參數用char*,Vector用char**,

動態內存要牢記誰申請誰釋放的原則。
 

posted on 2013-03-01 13:58 sheng 閱讀(6762) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


導航

<2025年12月>
30123456
78910111213
14151617181920
21222324252627
28293031123
45678910

統計

常用鏈接

留言簿(1)

隨筆檔案

收藏夾

同行

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久精品一区二区三区四区 | 久久久www免费人成黑人精品 | 噜噜噜91成人网| 欧美韩日一区二区| 亚洲免费在线观看视频| 久久久水蜜桃av免费网站| 日韩视频在线观看国产| 黄色小说综合网站| 日韩一区二区精品在线观看| 狠狠久久亚洲欧美| 模特精品在线| 另类天堂视频在线观看| 亚洲欧洲日韩综合二区| 久久精品国产精品亚洲综合| 亚洲一二区在线| 毛片精品免费在线观看| 久久久久久欧美| 国产精品午夜电影| 99国产精品久久久| 亚洲三级视频| 欧美高清视频一区二区| 亚洲色图制服丝袜| 亚洲一级二级| 黄色精品在线看| 欧美日韩美女一区二区| 亚洲国产清纯| 久久久久久91香蕉国产| 亚洲乱码一区二区| 在线一区观看| 欧美日韩视频在线| 久久精品在线免费观看| 麻豆精品传媒视频| 亚洲在线免费| 国产女人精品视频| 午夜精品久久久久99热蜜桃导演| 亚洲一区二区三区在线观看视频| 在线观看视频亚洲| 久久综合激情| 最新69国产成人精品视频免费| 亚洲精品久久久久久久久久久久久| 国产乱码精品1区2区3区| 性欧美1819性猛交| 久久综合伊人77777| 亚洲欧美日韩网| 亚洲精品美女免费| 伊人久久大香线| 国产久一道中文一区| 欧美日韩国语| 午夜精品999| 亚洲在线成人精品| 久久久久久高潮国产精品视| 亚洲一区二区免费| 一区二区三区高清在线观看| 最新亚洲电影| 亚洲国产精品激情在线观看| 欧美黄污视频| 久久婷婷久久| 99www免费人成精品| 91久久精品国产91性色tv| 免费看黄裸体一级大秀欧美| 一本久久a久久精品亚洲| 国产精品视频免费观看| 国产精品www| 久久频这里精品99香蕉| 久久精品一区二区三区四区 | 亚洲第一视频| 欧美日一区二区三区在线观看国产免| 亚洲一区中文| 亚洲综合好骚| 亚洲国产日日夜夜| 亚洲黄色免费| 亚洲精选成人| 宅男66日本亚洲欧美视频| 99精品视频免费观看| 一区二区三区国产| 欧美mv日韩mv国产网站app| 在线视频欧美日韩精品| 一区二区三区日韩欧美| 亚洲一区日本| 一区二区三区精品视频| 亚洲精品国产精品乱码不99按摩| 亚洲精品欧美日韩| 一本大道久久a久久精二百| 亚洲午夜久久久久久久久电影院 | 一本色道久久综合亚洲精品高清 | 国产亚洲a∨片在线观看| 久久嫩草精品久久久久| 久久久综合视频| 欧美 亚欧 日韩视频在线| 欧美人妖在线观看| 久久在线视频| 欧美精品一区三区| 国产精品盗摄久久久| 欧美大片免费看| 欧美专区在线观看| 亚洲一区二区三区免费观看| 亚洲欧美日韩综合一区| 久久久精品一区| 欧美丰满少妇xxxbbb| 国产精品国色综合久久| 好吊日精品视频| 日韩一区二区久久| 久久成人人人人精品欧| 亚洲在线免费| 久久午夜电影网| 久久国产免费| 欧美一级片久久久久久久| 在线亚洲欧美| 久久久精品日韩欧美| 亚洲国产清纯| 欧美一区免费视频| 欧美在线免费观看亚洲| 欧美激情一二区| 欧美日韩国产高清| 国模一区二区三区| 激情欧美日韩| 影音先锋成人资源站| 亚洲一区二区三区在线观看视频| 玖玖玖国产精品| 亚洲性xxxx| 小黄鸭精品aⅴ导航网站入口| 欧美黄色免费网站| 国产一区二区欧美日韩| 韩日欧美一区二区三区| 中日韩高清电影网| 午夜久久久久| 久久国产精品久久久| 亚洲精品欧美在线| 久热这里只精品99re8久| 国产精品女人久久久久久| 国产欧美精品va在线观看| 亚洲精品乱码| 美日韩精品免费观看视频| 亚洲主播在线播放| 欧美三区在线观看| 亚洲欧洲在线一区| 欧美aⅴ一区二区三区视频| 性欧美xxxx大乳国产app| 国产精品久久久久一区二区三区| 99视频精品全国免费| 欧美成人一区二区三区| 亚洲国产激情| 日韩一级黄色片| 欧美国产日韩一区二区| 亚洲国产人成综合网站| 亚洲无线一线二线三线区别av| 亚洲第一综合天堂另类专| 久久亚洲综合网| 亚洲电影免费观看高清完整版在线观看 | 亚洲国语精品自产拍在线观看| 久久久久这里只有精品| 国产一区二区中文字幕免费看| 午夜激情综合网| 亚洲午夜久久久久久久久电影院| 欧美激情国产日韩精品一区18| 亚洲激情网站| 亚洲国产小视频在线观看| 欧美91福利在线观看| 亚洲激情视频在线播放| 欧美激情成人在线视频| 亚洲一区二区三区四区中文| 欧美亚洲成人免费| 亚洲欧美国产毛片在线| 亚洲婷婷在线| 国产欧美另类| 久久久久久久久久久一区| 久久精品九九| 亚洲国产精品一区二区三区| 欧美激情一区三区| 欧美激情综合色综合啪啪| 一本色道婷婷久久欧美| 一本久久精品一区二区| 国产精品免费一区二区三区在线观看| 午夜视频在线观看一区二区三区 | 亚洲电影免费观看高清完整版| 免费中文日韩| 欧美激情亚洲国产| 亚洲性感激情| 欧美一区二区三区免费视| 尤物99国产成人精品视频| 亚洲高清一区二| 欧美性色aⅴ视频一区日韩精品| 欧美一级片在线播放| 久久精品99国产精品日本| 亚洲国产日韩在线一区模特| 亚洲精品一二三区| 国产精品一区二区三区久久 | 亚洲国产精品123| 欧美日韩三区| 久久久综合网站| 欧美黑人一区二区三区| 先锋影音国产精品| 免播放器亚洲| 午夜精品久久久久久99热软件| 久久精品网址| 亚洲视频每日更新| 久久久久国产精品一区三寸| 99热在线精品观看| 性色一区二区三区| 亚洲精品中文字|