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

OnTheWay2012
埋葬昨天的我,迎來重生的我!
posts - 15,  comments - 89,  trackbacks - 0
今天在寫代碼的時候發現了一個很有意思的錯誤,出錯的代碼如下。
加入你的電腦里定義了JAVA_HOME這個環境變量,你能看出程序的輸出結果是什么嗎?
 1#include <iostream>
 2#include <assert.h>
 3#include <map>
 4#include <string>
 5
 6using namespace std;
 7
 8bool GetEnvByKey(string const &strKey, string &strValue)
 9{
10    bool bRet = false;
11
12    assert(!strKey.empty());
13
14    size_t sizeValue = 0;
15    if(0 == getenv_s(&sizeValue, NULL, 0, strKey.c_str()))
16    {
17        strValue.reserve(sizeValue);
18        bRet = (0 == getenv_s(&sizeValue, const_cast<char*>(strValue.c_str()), sizeValue, strKey.c_str()));
19    }

20
21    return bRet;
22}

23
24int main( )
25{
26    string strKey("JAVA_HOME");
27    string strValue;
28    
29    if(GetEnvByKey(strKey, strValue))
30    {
31        cout<<strKey<<endl;
32        cout<<strValue<<endl;
33
34        map<stringstring> mapEnvInfo;
35        mapEnvInfo.insert(mapEnvInfo.end(), make_pair(strKey, strValue));
36
37        cout<<mapEnvInfo.size()<<endl;
38        cout<<mapEnvInfo.begin()->first<<endl;
39        cout<<mapEnvInfo.begin()->second<<endl;
40    }

41
42    return 0;
43}


先把你猜想的結果寫出來,然后再把代碼復制到VC里你試一下,看看到底輸出什么。你猜對了嗎?是不是輸出結果有點不太對?呵呵,經過調試代碼你發現錯誤了嗎?
錯誤就在
strValue.reserve(sizeValue);
bRet = (0 == getenv_s(&sizeValue, const_cast<char*>(strValue.c_str()), sizeValue, strKey.c_str()));
這兩行。
錯在這種使用strValue的方式,詳述如下:
strValue.reserve(sizeValue)這句代碼給strValue分配了sizeValue字節的內存,然后getenv_s又把從操作系統里取得的值賦給了strValue所代表的內存中,請注意strValue除了有指向字符串的內存數組,還有一個記錄該字符串中保存的字符數的一個變量_Mysize。上面的getenv_s執行后只是把字符串給賦值了,但是這個記錄字符串中字符個數的變量卻沒有賦值。下面的代碼需要使用這個變量才能進行輸出,而此時這個變量是0,所以strValue就沒有輸出。
經過上面的說明之后你肯定明白了上述代碼出錯的原因。為什么會出錯?是因為我們使用strValue的時候把它認為成了數組,所以就出錯了。本來strValue是一個對象,它具有各種方法,但是我們的使用方式強迫strValue脫下外衣,強迫它變成赤裸裸的字符串數組,這就是問題的發生根源。因此我們使用類對象的時候一定不能把類的外衣脫掉,讓類帶來的好處消失于無形。
posted on 2010-04-02 22:30 OnTheWay 閱讀(2819) 評論(9)  編輯 收藏 引用 所屬分類: C、C++

FeedBack:
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-02 22:54 | 楊帆
恩,明白,有收獲。3Q。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-02 23:10 | 楊帆
不對,我又看了看,我覺得樓主的理解似乎有問題,也請指教指教。

strValue.reserve(sizeValue);
bRet = (0 == getenv_s(&sizeValue, const_cast<char*>(strValue.c_str()), sizeValue, strKey.c_str()));

的確是這里有問題,問題的確出在const_cast<char*>(strValue.c_str())這個表達式上。

但是樓主想表達的意思是不是getenv_s()這個函數把strValue這個string類型“強行”當做的了一個字符串來進行處理,這個函數修改strValue的時候,僅僅修改了“一部分”?

我想應該是這樣,const_cast<char*>(strValue.c_str())是這么執行的:
1、strValue.c_str() 這里strValue返回了一個”臨時的“字符串,注意,是臨時的而且是const的,它應該是新開辟了一小段內存用以存儲這個C-Style字符串,而不是把strValue本身當做字符串給返回回去了。

2、使用const_cast<char *>將這個const 并且”臨時的“字符串進行了轉換,轉換成了 非const ,但仍然是臨時的字符串。

3、然后getenv_s()函數會對這個臨時非const的字符串進行一些操作。

4、隨著函數調用的結束,這個臨時的字符串被釋放掉了。

在上邊這個過程中,并沒有對strValue進行任何改變,也正因此在以后才什么都沒有輸出來。

和樓主不一樣的是,getenv_s()根本沒有對strValue進行任何操作。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-03 08:59 | OnTheWay
@楊帆
謝謝你的關注。
不過strValue.c_str()沒有返回什么臨時的指針。
因為c_str()函數返回的是一個char const *類型,這說明返回的指針是不可寫的。但是我為了讓返回的指針可寫,所以加了const_cast。這同時說明有const_cast的地方都存在潛在的錯誤。你可以在VC2005的debug版下看看程序具體的執行過程,在此過程中沒有創建臨時的字符串數組。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-03 09:24 | CBKID
@楊帆
我也感覺有點像你的看法,但是如果這樣的話可以先給strValue賦值,如果真的沒有進行任何操作的話結果就很顯眼了。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-03 13:06 | 楊帆
@OnTheWay
感謝樓主,恩,我驗證了一下,我的理解不正確,不好意思哈,也感謝從你這學到了東西,3Q。

我用下列代碼進行了驗證,在gcc下編譯通過。

#include <iostream>
#include <string>
using namespace std;

int main(){
string CppString("我是一個string啊,咿呀咿呀喲!");

//在這里用c_str()返回了一個CStyle字符串,保存在CStyleString上
char * CStyleString = const_cast< char *>( CppString.c_str() );

//首先驗證一下這個CStyleString的內容是不是正確
cout << CStyleString <<endl;

//然后對CStyleString進行一些改變。
cin >> CStyleString ;

//輸出改變后的CStyleString看看。
cout << CStyleString << endl;

//這時再輸出CppString的內容,它改變了!。
cout << CppString <<endl;
return 0;
}

程序運行如下:
我是一個string啊,咿呀咿呀喲!
我是一個CStyle String啊,咿呀咿呀喲! //這是我的輸入。
我是一個CStyle
我是一個CStyle ……&%¥*&…… //后邊一一堆亂碼,原因請見樓主的帖子。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-03 15:06 | 楊帆
進一步琢磨,我在http://www.cplusplus.com/上查了查c_str的描述,連接為:http: //www.cplusplus.com/reference/string/string/c_str/
描述如下:
const char* c_str ( ) const;

Get C string equivalent
Generates a null-terminated sequence of characters (c-string) with the same content as the string object and returns it as a pointer to an array of characters.

A terminating null character is automatically appended.

The returned array points to an internal location with the required storage space for this sequence of characters plus its terminating null-character, but the values in this array should not be modified in the program and are only granted to remain unchanged until the next call to a non-constant member function of the string object.


關鍵位最后一段:(湊乎著翻譯下,大家海涵哈,不要覺得它慘不忍睹哈,^_^)

該函數所返回的指針(數組)指向該字符串的內部(internal)位置,并且該位 置具有足夠存儲空間來存儲該字符序列以及表示結尾的空字符,但是返回數組中的值在程序不應該有任何修改并只應被傳值使用(這里的grant實在譯不好,望 大牛們指教),直到下一次調用該字符串對象的非常(non-const)成員函數。

這一段話再次驗證了樓主的說法。


  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-03 19:44 | OnTheWay
@楊帆
感謝你的持續關注。
你的鉆研精神值得敬佩。  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-05 08:18 | 99書城
福建省地方你們是地方  回復  更多評論
  
# re: 不要把類的外衣脫下來,讓類的美麗消失于無形
2010-04-13 02:29 | 欲三更
代碼沒看,但是要如lz所說的話,好無厘頭的錯誤啊。。。  回復  更多評論
  

<2010年11月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用鏈接

留言簿(4)

隨筆分類

隨筆檔案

友情連接

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久久久综合一区二区三区| 日韩一区二区福利| 亚洲视频一区二区在线观看| 午夜视黄欧洲亚洲| 亚洲永久免费精品| 久久精品盗摄| 久久久噜噜噜久久人人看| 欧美日韩视频专区在线播放| 欧美激情一区二区三区在线视频观看| 久久久久中文| 国产亚洲制服色| 美脚丝袜一区二区三区在线观看 | 欧美日本精品| 欧美一区二区三区在| 欧美成人精精品一区二区频| 亚洲视频axxx| 亚洲第一精品福利| 一区二区三区国产| 米奇777超碰欧美日韩亚洲| 99re6热在线精品视频播放速度| 国产精品久久久久高潮| 欧美第一黄色网| 午夜日韩在线观看| 好吊色欧美一区二区三区视频| 午夜在线视频观看日韩17c| 亚洲国产欧美不卡在线观看| 久久视频在线看| 久久激情一区| 久久久久9999亚洲精品| 香蕉久久一区二区不卡无毒影院| 一本色道88久久加勒比精品| 亚洲国产欧美久久| 亚洲毛片在线看| 欧美大片一区二区三区| 老司机精品导航| 欧美高清在线一区| 欧美高清不卡| 欧美日韩视频在线一区二区观看视频| 欧美 日韩 国产 一区| 久久琪琪电影院| 亚洲欧美日韩视频二区| 亚洲欧洲一区二区在线播放| 亚洲国产美女精品久久久久∴| 亚洲高清视频在线| 亚洲精品免费在线观看| 亚洲视频在线观看网站| 亚洲与欧洲av电影| 蜜臀av国产精品久久久久| 欧美bbbxxxxx| 亚洲免费在线观看视频| 蜜臀久久99精品久久久久久9 | 久久精品视频亚洲| 欧美—级高清免费播放| 国产精品sss| 亚洲专区一区二区三区| 亚洲一二三区视频在线观看| 校园春色综合网| 欧美日韩一区高清| 在线观看国产精品网站| 亚洲欧美另类中文字幕| 欧美大学生性色视频| 亚洲欧美日韩网| 欧美视频官网| 亚洲免费播放| 欧美1区2区| 老司机一区二区| 亚洲在线中文字幕| 欧美日韩一区在线| 亚洲美女中文字幕| 亚洲成色精品| 欧美一区在线直播| 亚洲视频在线观看视频| 久久丁香综合五月国产三级网站| 亚洲在线免费视频| 国产精品国产三级国产普通话三级 | 一区二区欧美国产| 欧美日韩亚洲一区二区三区四区| 亚洲韩国青草视频| 亚洲日本中文字幕| 亚洲人成在线观看| 欧美国产日韩视频| 日韩一区二区福利| 一本色道久久综合亚洲精品高清| 欧美久久一区| 欧美一区二区三区另类| 亚洲欧美不卡| 91久久黄色| 欧美久久久久| 亚洲欧美激情四射在线日 | 亚洲国产日韩欧美在线图片| 亚洲一区二区三区乱码aⅴ| 国产精品久久久久7777婷婷| 欧美a级理论片| 久久亚洲欧美国产精品乐播| 国产区亚洲区欧美区| 国产一级揄自揄精品视频| 在线一区二区三区做爰视频网站| 久久大香伊蕉在人线观看热2| 欧美亚洲综合另类| 国产亚洲欧美日韩日本| 亚洲影院免费观看| 久久精品国产亚洲一区二区三区| 国产精品福利网| 亚洲午夜视频在线| 一区二区久久久久| 在线观看国产日韩| 欧美在线播放一区| 免费在线日韩av| 亚洲激情视频网| 欧美成人有码| 亚洲欧洲日韩在线| a91a精品视频在线观看| 欧美日韩中文| 亚洲欧美日本精品| 亚洲自拍三区| 国产亚洲一区二区三区在线观看| 亚洲欧美视频| 黄色欧美成人| 麻豆91精品91久久久的内涵| 欧美黄色片免费观看| 日韩一级大片在线| 国产精品视频久久久| 性感少妇一区| 亚洲国产成人在线| 99亚洲伊人久久精品影院红桃| 亚洲黑丝在线| 国产精品亚洲综合色区韩国| 欧美寡妇偷汉性猛交| 亚洲精品在线免费观看视频| 久久福利精品| 久久国产日韩| 欧美一区二区三区男人的天堂| 亚洲美女视频网| 日韩视频中文| 亚洲国产精品成人综合色在线婷婷| 国产欧美高清| 欧美久久视频| 欧美激情精品久久久久| 欧美在线视频不卡| 亚洲一区视频在线| 一个色综合av| 一区二区三区欧美激情| 亚洲国产精品成人精品| 国产亚洲欧美aaaa| 国产精品久久久久久久9999| 久久成人国产精品| 在线观看国产精品网站| 国产在线麻豆精品观看| 狠狠色噜噜狠狠色综合久| 国产精品女主播| 国产精品久久一区主播| 欧美体内she精视频| 国产精品美女久久久免费| 欧美精品一区二区三区一线天视频| 伊人成年综合电影网| 久久精品国产v日韩v亚洲| 性欧美大战久久久久久久久| 亚洲一区二区精品| 久久成人精品视频| 久久久国产一区二区| 亚洲福利专区| 亚洲一级影院| 久热精品视频在线观看| 国产精品日韩欧美大师| 亚洲国产小视频| 免费观看在线综合| 欧美一区二区三区婷婷月色 | 亚洲黄色成人久久久| 久久麻豆一区二区| 亚洲第一在线| 欧美在线综合视频| 欧美激情1区2区3区| 国产日韩在线亚洲字幕中文| 日韩亚洲精品电影| 亚洲高清视频一区二区| 亚洲小视频在线观看| 在线观看中文字幕亚洲| 国产精品观看| 一本色道久久综合| 夜夜嗨av一区二区三区中文字幕 | 欧美国产视频在线| 亚洲午夜精品17c| 欧美日韩一区在线视频| 亚洲激情影视| 免费国产一区二区| 欧美一区二区三区在线免费观看| 亚洲综合日韩在线| 一区二区三区四区国产| 午夜伦理片一区| 久久国产精品99精品国产| 国产日本欧美在线观看| 毛片基地黄久久久久久天堂| 另类春色校园亚洲| 亚洲国产另类 国产精品国产免费| 欧美日韩伦理在线| 一区二区三区欧美在线| 亚洲欧美另类中文字幕| 国产在线拍揄自揄视频不卡99| 噜噜噜躁狠狠躁狠狠精品视频| 暖暖成人免费视频|