• <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>

            woaidongmao

            文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
            隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
            數據加載中……

            猜猜看,id變成9了嗎?

            #include "stdafx.h"
            #include "stdio.h"
            struct TestStr
            {
                int id;
                void SetId(int id_new){id = id_new;} 
            };

            int main(int argc, char* argv[])
            {
                const TestStr ts = {1};
                const TestStr* p_ts = &ts;
                ((TestStr)(*p_ts)).SetId(9);
                printf("TestStr::id = %d\r\n",ts.id);
                return 0;
            }


            id還是等于1,不信試試,順便替換成下面這一句再試試看
            ((TestStr&)(*p_ts)).SetId(9);


            另外還有一個奇怪的問題:
            ((TestStr)(*p_ts)).SetId(9);      //編譯通過
            ((TestStr)(*p_ts)).id = 9;         //編譯不通過
            (&((TestStr)(*p_ts)))->id = 9; //編譯通過

            誰可以解釋一下原因?不知。

            posted on 2008-04-01 22:03 肥仔 閱讀(1639) 評論(10)  編輯 收藏 引用 所屬分類: C++ 基礎

            評論

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            ((TestStr)(*p_ts)).SetId(9); 如果變成(TestStr(*p_ts)).SetId(9);
            這其實相當于調用了TestStr(const TestStr&)生成了一個TestStr&的臨時變量。
            按理來說這三個應該都可以,確實在gcc 4.1.3下可以全部編譯通過的。
            這個可能和編譯器有關,在編譯((TestStr)(*p_ts)).id = 9; 時生成的臨時變量可能為const的,所以編譯不過。
            2008-04-01 23:18 | www

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            @www
            //((TestStr)(*p_ts)).SetId(9); 如果變成(TestStr(*p_ts)).SetId(9);
            ((TestStr)(*p_ts)).SetId(9); 和(TestStr(*p_ts)).SetId(9); 是一樣的,//上面寫錯了
            2008-04-01 23:20 | www

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            C++標準規定,類性轉換表達式的目標類型為引用時,結果為l-value;否則為r-value

            對于built-in 類型,r-value是不可修改的;而對于user-defined類型,r-value在某些情況下是允許修改的

            GCC編譯沒問題,m$對標準的支持存在問題吧
            2008-04-02 00:01 | 嘯天豬

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            關于結果等于1的問題,((TestStr)(*p_ts)) 新生成了一個對象,因為p_ts指向的是const
            2008-04-02 09:27 | hsen

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            ((TestStr)(*p_ts)).SetId(9); //編譯通過
            ((TestStr)(*p_ts)).id = 9; //編譯不通過
            (&((TestStr)(*p_ts)))->id = 9; //編譯通過


            我在VS2005下:
            正如作者所說中間的是編不過的,error,說是l-value不能賦值。
            但是能編過的2個結果都還是1,沒有修改了原來的值,因為在類型轉化的時候都調用了拷貝構造函數,從新生成一個對象你修改的是拷貝后的臨時對象。(你可以寫拷貝構造函數測試一下)


            所以同意:
            這其實相當于調用了TestStr(const TestStr&)生成了一個TestStr&的臨時變量。
            C++標準規定,類性轉換表達式的目標類型為引用時,結果為l-value;否則為r-value。


            2008-04-02 10:02 | 夢在天涯

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            類型轉換(非引用)必然會產生一個臨時對象。所有的一切問題都可以歸結到這里:
            ((TestStr)(*p_ts)).SetId(9);//產生non-const臨時對象,SetId()操作的是臨時對象,因此原對象不變
            ((TestStr&)(*p_ts)).SetId(9);//去掉原對象的const。可以寫成const_cast<TestStr&>(*p_ts)).SetId(9)
            ((TestStr)(*p_ts)).id = 9; //GCC可以編譯
            (&((TestStr)(*p_ts)))->id = 9; //編譯通過
            2008-04-02 13:57 | raof01

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            @夢在天涯
            這個有道理,有人驗證過嗎?
            2008-04-02 16:16 | foobar

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            你看下你的反匯編代碼就知道了。
            編譯的時候,常量在允許的情況下編譯器會幫你替換成立即數(因為立即尋址比較快)。
            所以你的printf("TestStr::id = %d\r\n",ts.id)在編譯的時候會當成
            printf("TestStr::id = %d\r\n",1)編譯。
            但是ts所在內存的內容是的確已經改變了。
            你把(*p_ts)打出來看看就知道了。
            對于指針,編譯器是沒有辦法在編譯的時候替換成立即數的。
            2008-04-02 19:52 | -.-:

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            把void SetId(int id_new)的參數改成引用或指針類型試試?


            2008-04-08 17:45 | ww

            # re: 猜猜看,id變成9了嗎?  回復  更多評論   

            ((TestStr)(*p_ts)).SetId(9); //編譯通過
            ((TestStr)(*p_ts)).id = 9; //編譯不通過
            (&((TestStr)(*p_ts)))->id = 9; //編譯通過


            1,3 大家都講了,這個很好理解,實際上是生成了臨時的對象。編譯通過但是不會修改原來的變量。
            2 可以通過這個例子來理解

            class A;
            Func(A& p) {...}
            main()
            {
            Func(A());//編譯錯誤,因為臨時對象必須是const&
            }

            如果Func(A const&p) {A &c =(A&)p;c.xxx}這是合法的。
            所以3,就類似與后面這個Func 作的事情。

            臨時對象不能修改,這是c++標準規定的,在侯捷書More C++ Exception里面提到過的。至于g++可以編譯通過,G++在默認拷貝時候構造的對象多于1個造成的。大家可以在析構函數里輸出就可以看出來了。
            2008-12-20 23:05 | 楊成
            久久精品国产亚洲AV香蕉| 久久精品亚洲一区二区三区浴池| 久久精品国产精品亚洲毛片| 久久中文骚妇内射| 99精品久久久久久久婷婷| 久久露脸国产精品| 999久久久无码国产精品| 久久国产精品免费| 久久91精品久久91综合| 狼狼综合久久久久综合网| 久久亚洲国产欧洲精品一 | 精品一二三区久久aaa片| 国产精品毛片久久久久久久| 中文成人久久久久影院免费观看| 四虎国产永久免费久久| 久久久无码精品亚洲日韩蜜臀浪潮| 国产精品久久亚洲不卡动漫| 少妇久久久久久被弄到高潮| 久久AⅤ人妻少妇嫩草影院| 思思久久99热只有频精品66| 亚洲伊人久久综合影院| 97久久综合精品久久久综合| 日韩十八禁一区二区久久| 久久精品国产99国产精偷 | 99久久精品免费国产大片| 99久久无色码中文字幕人妻| 久久久精品人妻一区二区三区蜜桃| 亚洲一区中文字幕久久| 99久久国产宗和精品1上映| 久久伊人精品青青草原日本| 亚洲精品国产成人99久久| 国内精品久久久久影院免费| 色8久久人人97超碰香蕉987| 久久精品一区二区三区AV| 亚洲日本久久久午夜精品| 四虎影视久久久免费观看| 久久久精品久久久久久 | 亚洲国产精品一区二区久久| 狠狠狠色丁香婷婷综合久久俺| 久久精品国产清高在天天线| 久久天天躁狠狠躁夜夜96流白浆|