• <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>
            OnTheWay2012
            埋葬昨天的我,迎來(lái)重生的我!
            posts - 15,  comments - 89,  trackbacks - 0
            今天在寫(xiě)代碼的時(shí)候發(fā)現(xiàn)了一個(gè)很有意思的錯(cuò)誤,出錯(cuò)的代碼如下。
            加入你的電腦里定義了JAVA_HOME這個(gè)環(huán)境變量,你能看出程序的輸出結(jié)果是什么嗎?
             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}


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

            FeedBack:
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-02 22:54 | 楊帆
            恩,明白,有收獲。3Q。  回復(fù)  更多評(píng)論
              
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-02 23:10 | 楊帆
            不對(duì),我又看了看,我覺(jué)得樓主的理解似乎有問(wèn)題,也請(qǐng)指教指教。

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

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

            但是樓主想表達(dá)的意思是不是getenv_s()這個(gè)函數(shù)把strValue這個(gè)string類(lèi)型“強(qiáng)行”當(dāng)做的了一個(gè)字符串來(lái)進(jìn)行處理,這個(gè)函數(shù)修改strValue的時(shí)候,僅僅修改了“一部分”?

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

            2、使用const_cast<char *>將這個(gè)const 并且”臨時(shí)的“字符串進(jìn)行了轉(zhuǎn)換,轉(zhuǎn)換成了 非const ,但仍然是臨時(shí)的字符串。

            3、然后getenv_s()函數(shù)會(huì)對(duì)這個(gè)臨時(shí)非const的字符串進(jìn)行一些操作。

            4、隨著函數(shù)調(diào)用的結(jié)束,這個(gè)臨時(shí)的字符串被釋放掉了。

            在上邊這個(gè)過(guò)程中,并沒(méi)有對(duì)strValue進(jìn)行任何改變,也正因此在以后才什么都沒(méi)有輸出來(lái)。

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

            我用下列代碼進(jìn)行了驗(yàn)證,在gcc下編譯通過(guò)。

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

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

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

            //首先驗(yàn)證一下這個(gè)CStyleString的內(nèi)容是不是正確
            cout << CStyleString <<endl;

            //然后對(duì)CStyleString進(jìn)行一些改變。
            cin >> CStyleString ;

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

            //這時(shí)再輸出CppString的內(nèi)容,它改變了!。
            cout << CppString <<endl;
            return 0;
            }

            程序運(yùn)行如下:
            我是一個(gè)string啊,咿呀咿呀喲!
            我是一個(gè)CStyle String啊,咿呀咿呀喲! //這是我的輸入。
            我是一個(gè)CStyle
            我是一個(gè)CStyle ……&%¥*&…… //后邊一一堆亂碼,原因請(qǐng)見(jiàn)樓主的帖子。  回復(fù)  更多評(píng)論
              
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-03 15:06 | 楊帆
            進(jìn)一步琢磨,我在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.


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

            該函數(shù)所返回的指針(數(shù)組)指向該字符串的內(nèi)部(internal)位置,并且該位 置具有足夠存儲(chǔ)空間來(lái)存儲(chǔ)該字符序列以及表示結(jié)尾的空字符,但是返回?cái)?shù)組中的值在程序不應(yīng)該有任何修改并只應(yīng)被傳值使用(這里的grant實(shí)在譯不好,望 大牛們指教),直到下一次調(diào)用該字符串對(duì)象的非常(non-const)成員函數(shù)。

            這一段話(huà)再次驗(yàn)證了樓主的說(shuō)法。


              回復(fù)  更多評(píng)論
              
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-03 19:44 | OnTheWay
            @楊帆
            感謝你的持續(xù)關(guān)注。
            你的鉆研精神值得敬佩。  回復(fù)  更多評(píng)論
              
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-05 08:18 | 99書(shū)城
            福建省地方你們是地方  回復(fù)  更多評(píng)論
              
            # re: 不要把類(lèi)的外衣脫下來(lái),讓類(lèi)的美麗消失于無(wú)形
            2010-04-13 02:29 | 欲三更
            代碼沒(méi)看,但是要如lz所說(shuō)的話(huà),好無(wú)厘頭的錯(cuò)誤啊。。。  回復(fù)  更多評(píng)論
              

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            常用鏈接

            留言簿(4)

            隨筆分類(lèi)

            隨筆檔案

            友情連接

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            青草影院天堂男人久久| 综合久久精品色| 久久精品国产精品亚洲人人| 午夜精品久久久久久| 久久精品中文无码资源站| 国产成人久久激情91| 亚洲级αV无码毛片久久精品| 青青热久久综合网伊人| 97精品久久天干天天天按摩| 久久精品国产色蜜蜜麻豆| 欧美噜噜久久久XXX| 久久久精品国产免大香伊| 亚洲成色999久久网站| 无码超乳爆乳中文字幕久久| 久久精品综合网| 99久久国产热无码精品免费久久久久| 国产精品久久午夜夜伦鲁鲁| 久久国产色av免费看| 精品久久久无码中文字幕| 51久久夜色精品国产| 亚洲欧美伊人久久综合一区二区 | 婷婷久久五月天| 久久se这里只有精品| 久久最新精品国产| 91精品国产9l久久久久| 99精品国产在热久久无毒不卡| 色诱久久久久综合网ywww| 无码乱码观看精品久久| 久久久久国产精品三级网| 欧美伊香蕉久久综合类网站| 久久久久久综合一区中文字幕 | 久久人搡人人玩人妻精品首页 | 久久久久久国产a免费观看黄色大片| 午夜精品久久久久久影视777| 久久综合久久性久99毛片| 国产精品免费久久久久影院| 久久久青草青青国产亚洲免观| 国产精品久久久久久久app | 97久久精品人人澡人人爽| 久久99精品久久久久久水蜜桃| 亚洲精品第一综合99久久|