• <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>
            隨筆-5  評論-31  文章-0  trackbacks-0
            今年要開始找工作了,本著積累經驗的目的,跑去做了下MTK的筆試題,筆試的內容主要是C++。
            因為開發中一直使用C++,而且對C++里的高級特性:面向對象,模板等都比較熟悉,還沒事喜歡研究下STL,BOOST,所以對自己的C++水平比較自信,因此事先也沒做任何準備,就直接去筆試了。本來筆試完了后覺得題目蠻簡單的,但是本著認真學習的態度回來后把題目都上機試驗了下,結果一下就悲劇了,錯的體無完服啊。。。
            總結了一下:
               1。認真對待,不要小看了筆試題目:做題的時候心想這些筆試題目都很簡單啊,很多題目都是掃了一眼就立即寫出了答案,結果回來后才發現這些題目都設置了陷阱,讓你掉進去就出不來了。
               2。C++基礎不夠扎實。枉我還一天到晚的研究C++的高級特性,結果很多基礎的知識卻都是一知半解。
            特將此次筆試的一些心得和體會記錄于此,好提醒自己。下面主要分析幾個我做錯的題目。題目并非與原題完全一致。
            題目一:
            int a=10,b=6;
            cout
            <<a+b<<" "<<a++<<" "<<b++

            請說出上述語句的執行結果。
            很多人看過這段代碼后估計都會直接就寫上了 16 10 6 這樣的結果吧,但上機實驗的輸出結果是: 18 10 6
            為什么會出現這樣的結果,下面是我的分析過程,如果有不對的地方請大家指正。
            為了跟蹤代碼的執行步驟,我設計了一個類X,這個類是對int的模擬,行為方面與int基本一致,除了會打印出一些幫助我們理解的信息,代碼如下:

            class X
            {
            public:
                X(){cout
            <<"default construct"<<endl;}
                X(
            int a):i(a){ cout<<"construct "<<i<<endl;}
                
            ~X(){ cout<<"desconstruct "<<i<<endl;}
                X(
            const X& x):i(x.i)
                {
                    cout
            <<"copy construct "<<i<<endl;
                }
                X
            & operator++()
                {
                    cout
            <<"operator ++(pre) "<<i<<endl;
                    
            ++i;
                    
            return *this;
                }
                
            const X operator++(int)
                {
                    cout
            <<"operator ++(post) "<<i<<endl;
                    X x(
            *this);
                    
            ++i;
                    
            return x;
                }
                X
            & operator=(int m)
                {
                    cout
            <<"operator =(int)"<<endl;
                    i 
            = m;
                    
            return *this;
                }
                X
            & operator=(const X& x)
                {
                    cout
            <<"operator =(X)"<<endl;
                    i
            =x.i;
                    
            return *this;
                }
                
            /////////////////////////
                friend ostream& operator<<(ostream& os,const X& x)
                {
                    os
            <<x.i;
                    
            return os;
                }
                friend X 
            operator+(const X& a,const X& b)
                {
                    cout
            <<"operator +"<<endl;
                    return X(a.i+b.i);
                }
                
            //////////////////////////
            public:
                
            int i;
            };

            然后執行以下代碼:

                X a(10),b(6);
                cout
            <<"sum:" <<a+b<<" a:"<<a++<<" b:"<<b++<<endl;

            使用GCC4。5編譯后,代碼的執行結果如下:

            construct 10
            construct 6
            operator ++(post) 6
            copy construct 6
            operator ++(post) 10
            copy construct 10
            operator +
            construct 18
            sum:18 a:10 b:6
            desconstruct 18
            desconstruct 10
            desconstruct 6
            desconstruct 7
            desconstruct 11
            我們來簡單分析下這個執行過程:

            construct 10
            construct 6  //這兩行輸出對應于 X a(10),b(6); 

            operator ++(post) 6
            copy construct 6 //表明首先執行了  cout<<"sum:" <<a+b<<" a:"<<a++<<" b:"<<b++<<endl;這句中的 b++這個表達式,
                                          b++這個表達式返回了一個值為6的臨時對象,而b本身則變成了7。
            operator ++(post) 10
            copy construct 10  //這句的分析同上

            operator +
            construct 18 //對應于表達式 a+b ,可以看到,此時的a和b已經變成了11和7。表達式返回了一個值為18的臨時對象。

            sum:18 a:10 b:6 //輸出的結果,從結果可以看出,實際上打印出的值分別為 a+b,a++和b++三個表達式所返回的臨時變量。

            desconstruct 18 //a+b 表達式返回的臨時變量的析構
            desconstruct 10 //a++ 表達式返回的臨時變量的析構
            desconstruct 6 //b++表達式返回的臨時變量的析構
            desconstruct 7 //變量a 的析構
            desconstruct 11  //變量b的析構

            真相大白了。為什么編譯器會這樣來編譯這個表達式呢?
            下面2樓的夜風同學給出了正確答案。。為了不誤導后面的同學,特此編輯掉。。

            上述實驗的環境均為GCC4。5  據同學說VS2010執行的結果在DEBUG下和RELEASE下居然分別為:16 10 6 和18 10 6,不過我沒有去驗證過,有興趣的同學可以去驗證并分析一下。
            做這樣一道題還是讓我收獲很多,鞏固了C++的基礎。
            今天就寫道這里,后面有時間會陸續放出對其他“陷阱”題目的分析。
            (未完待續)

            posted on 2011-08-13 17:30 江浸月 閱讀(3257) 評論(19)  編輯 收藏 引用

            評論:
            # re: 做MTK筆試的總結(一)--C++ 運算符優先級 2011-08-13 19:50 | pansunyou
            確實輸出不同的結果:

            [VS2008] cl.exe 15.00.21022.08 DEBUG輸出16 10 6
            [VS2008] cl.exe 15.00.21022.08 RELEASE輸出18 10 6
            [UNIX SCO] CC 3.1 09/28/99 DEBUG/RELEASE輸出18 10 6
            gcc version 4.1.2 20070115 DEBUG/RELEASE輸出18 10 6
              回復  更多評論
              
            # re: 做MTK筆試的總結(一)--C++ 運算符優先級 2011-08-13 21:00 | 夜風
            <<在同一語句中連續使用,其實本質上是函數的復合調用
            cout<<a+b<<" "<<a++<<" "<<b++;
            本質上是
            operator<<(operator<<(operator<<(cout,a+b),a++),b++)
            由于c函數參數傳遞順序是從右至左,所以參數的計算次序是:
            b++ //7
            a++ //11
            a+b //18
            cout<<18
            cout<<11 //應該是10,因為已經先入棧了
            cout<<7 //應該是6  回復  更多評論
              
            # re: 做MTK筆試的總結(一)--C++ 運算符優先級 2011-08-13 21:19 | right
            盡管這種問題有確定的答案,但是實際上永遠不要寫這種代碼。  回復  更多評論
              
            # re: 做MTK筆試的總結(一)--C++ 運算符優先級 2011-08-13 21:47 | 江浸月
            @夜風
            原來如此,受教了。不過最后應該是
            b++ //7
            a++ //11
            a+b //18
            cout<<18
            cout<<10
            cout<<6
            才對。  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-13 23:31 | 瘋狂的面包
            http://blog.csdn.net/luciferisnotsatan/article/details/6456696
            我來給大家一個鏈接 很好解析這個問題  回復  更多評論
              
            # re: 做MTK筆試的總結(一)[未登錄] 2011-08-14 16:03 | a
            題目也太水了吧。
            第一題 就是 UB.
              回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-14 19:04 | 俺不是壞人
            未定義行為,結果多少都是正確的。  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-14 19:38 | 狂接
            UB吧  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-14 21:34 | ccsdu2009
            這樣的題目 只能說蛋疼  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-15 09:38 | 各個
            這種題純粹就是坑爹的,出這種題的人就是一SB  回復  更多評論
              
            # re: 做MTK筆試的總結(一)[未登錄] 2011-08-15 12:39 | Chipset
            可以看出出題的人根本不懂C/C++,這樣的公司不去也罷。  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-15 22:58 | 夜風
            @江浸月
            哦,對的,10和6已經入棧了  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-15 23:07 | 夜風
            @Chipset
            不見的,有可能題目的用意在于考察是否理解<<操作符的函數形式,還有函數參數入棧順序,如果這樣理解,還是比較有技術含量的  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-15 23:13 | 夜風
            @夜風
            如果不理解,還真有可能出現大問題,我曾經就遇到過一個問題,后來看匯編代碼時才回憶起<<的二元函數形式  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-17 00:32 | 艸,蛋疼
            在VC10上release才是16 10 6!






































              回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-17 12:40 | dougulf
            正確答案應該是:未定義行為!
            還問這種題目,如果面試的人期待的不是“未定義行為”的答案的話,只能說面試的人是在拿自己曾經出過的丑事讓別人出丑而已。
              回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-29 15:02 | 夢在天涯
            MKT現在還在找啊?  回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-08-29 15:52 | song
            在HP C/aC++ B3910B A.06.15
            試驗了下是16 10 6
              回復  更多評論
              
            # re: 做MTK筆試的總結(一) 2011-09-30 11:36 | chencj
            1、C函數參數的計算順序是不確定的,可以在C99文檔中找到(72頁),因此不同的編譯器處理可能會不一樣
            The order of evaluation of the function designator, the actual arguments, and
            subexpressions within the actual arguments is unspecified, but there is a sequence point
            before the actual call.
            2、這里的operator<<不是全局的二元操作符,而是ostream的成員運算符,一般是要重載自定類的<<操作符時,才會定義全局的二元操作符,而這里的cout是ostream。
            即cout<<a+b<<a++<<b++; 本質上是:
            ((cout.operator<<(a+b)).operator<<(a++)).operator<<(b++);
            3、二樓的夜風是正解
            以上供探討。  回復  更多評論
              
            久久精品国产乱子伦| 亚洲狠狠久久综合一区77777 | 亚洲人成电影网站久久| AV无码久久久久不卡蜜桃| 久久精品国产99久久久香蕉| 无码人妻久久一区二区三区免费丨 | 99久久精品免费观看国产| 久久中文精品无码中文字幕| 国产欧美一区二区久久| 久久久久亚洲Av无码专| 国产精品成人久久久| 久久久青草青青国产亚洲免观| 久久人人爽人人爽人人片AV高清| 久久成人国产精品| 亚洲午夜无码AV毛片久久| 激情综合色综合久久综合| 99久久精品影院老鸭窝| 久久香蕉综合色一综合色88| 久久精品国产91久久麻豆自制| 色婷婷久久久SWAG精品| 久久久久久综合网天天| 国产亚州精品女人久久久久久| 色偷偷偷久久伊人大杳蕉| 九九久久99综合一区二区| 日韩亚洲国产综合久久久| 久久精品九九亚洲精品天堂| 伊人久久大香线蕉av不卡| 亚洲午夜无码久久久久小说| 狠狠久久综合伊人不卡| 久久免费高清视频| 久久777国产线看观看精品| 亚洲国产精品无码久久| 久久精品日日躁夜夜躁欧美| 蜜臀久久99精品久久久久久| 国产精品女同一区二区久久| 九九精品久久久久久噜噜| 久久久久99精品成人片| 久久久久久A亚洲欧洲AV冫| 天天综合久久一二三区| 亚洲欧美久久久久9999| 亚洲愉拍99热成人精品热久久 |