• <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>
            asm, c, c++ are my all
            -- Core In Computer
            posts - 139,  comments - 123,  trackbacks - 0
            mutable關鍵字

            關鍵字mutable是C++中一個不常用的關鍵字,他只能用于類的非靜態和非常量數據成員
            我們知道一個對象的狀態由該對象的非靜態數據成員決定,所以隨著數據成員的改變,
            對像的狀態也會隨之發生變化!

            如果一個類的成員函數被聲明為const類型,表示該函數不會改變對象的狀態,也就是
            該函數不會修改類的非靜態數據成員.但是有些時候需要在該類函數中對類的數據成員
            進行賦值.這個時候就需要用到mutable關鍵字了

            例如:
            class Demo
            {
            public:
            ??? Demo(){}
            ??? ~Demo(){}
            public:
            ??? bool getFlag() const
            ??? {
            ??????? m_nAccess++;
            ??????? return m_bFlag;
            ??? }
            private:
            ??? int? m_nAccess;
            ??? bool m_bFlag;
            };

            int main()
            {
            ??? return 0;
            }

            編譯上面的代碼會出現 error C2166: l-value specifies const object的錯誤
            說明在const類型的函數中改變了類的非靜態數據成員.

            這個時候需要使用mutable來修飾一下要在const成員函數中改變的非靜態數據成員
            m_nAccess,代碼如下:

            class Demo
            {
            public:
            ??? Demo(){}
            ??? ~Demo(){}
            public:
            ??? bool getFlag() const
            ??? {
            ??????? m_nAccess++;
            ??????? return m_bFlag;
            ??? }
            private:
            ??? mutable int? m_nAccess;
            ??? bool m_bFlag;
            };

            int main()
            {
            ??? return 0;
            }

            這樣再重新編譯的時候就不會出現錯誤了!

            ?

            ?

            volatile關鍵字

            volatile是c/c++中一個鮮為人知的關鍵字,該關鍵字告訴編譯器不要持有變量的臨時拷貝,它可以適用于基礎類型
            如:int,char,long......也適用于C的結構和C++的類。當對結構或者類對象使用volatile修飾的時候,結構或者
            類的所有成員都會被視為volatile.

            使用volatile并不會否定對CRITICAL_SECTION,Mutex,Event等同步對象的需要
            例如:
            int i;
            i = i + 3;
            無論如何,總是會有一小段時間,i會被放在一個寄存器中,因為算術運算只能在寄存器中進行。一般來說,volatitle
            關鍵字適用于行與行之間,而不是放在行內。

            我們先來實現一個簡單的函數,來觀察一下由編譯器產生出來的匯編代碼中的不足之處,并觀察volatile關鍵字如何修正
            這個不足之處。在這個函數體內存在一個busy loop(所謂busy loop也叫做busy waits,是一種高度浪費CPU時間的循環方法)

            void getKey(char* pch)
            {
            ?while (*pch == 0)
            ??;
            }

            當你在VC開發環境中將最優化選項都關閉之后,編譯這個程序,將獲得以下結果(匯編代碼)
            ;?????? while (*pch == 0)
            $L27
            ?; Load the address stored in pch
            ?mov eax, DWORD PTR _pch$[ebp]
            ?; Load the character into the EAX register
            ?movsx eax, BYTE PTR [eax]
            ?; Compare the value to zero
            ?test eax, eax
            ?; If not zero, exit loop
            ?jne $L28
            ?;
            ?jmp $L27
            $L28
            ;}

            這段沒有優化的代碼不斷的載入適當的地址,載入地址中的內容,測試結果。效率相當的低,但是結果非常準確

            現在我們再來看看將編譯器的所有最優化選項開關都打開以后,重新編譯程序,生成的匯編代碼,和上面的代碼
            比較一下有什么不同
            ;{
            ?; Load the address stored in pch
            ?mov eax, DWORD PTR _pch$[esp-4]
            ?; Load the character into the AL register
            ?movsx al, BYTE PTR [eax]
            ;?while (*pch == 0)
            ?; Compare the value in the AL register to zero
            ?test al, al
            ?; If still zero, try again
            ?je SHORT $L84
            ?;
            ;}

            從代碼的長度就可以看出來,比沒有優化的情況要短的多。需要注意的是編譯器把MOV指令放到了循環之外。這在
            單線程中是一個非常好的優化,但是,在多線程應用程序中,如果另一個線程改變了變量的值,則循環永遠不會
            結束。被測試的值永遠被放在寄存器中,所以該段代碼在多線程的情況下,存在一個巨大的BUG。解決方法是重新
            寫一次getKey函數,并把參數pch聲明為volatile,代碼如下:

            void getKey(volatile char* pch)
            {
            ?while (*pch == 0)
            ??;
            }

            這次的修改對于非最優化的版本沒有任何影響,下面請看最優化后的結果:

            ;{
            ?; Load the address stored in pch
            ?mov eax, DWORD PTR _pch$[esp-4]
            ;?????? while (*pch == 0)
            $L84:
            ?; Directly compare the value to zero
            ?cmp BYTE PTR [eax], 0
            ?; If still zero, try again
            ?je SHORT $L84
            ?;
            ;}

            這次的修改結果比較完美,地址不會改變,所以地址聲明被移動到循環之外。地址內容是volatile,所以每次循環
            之中它不斷的被重新檢查。

            把一個const volatile變量作為參數傳遞給函數是合法的。如此的聲明意味著函數不能改變變量的值,但是變量的
            值卻可以被另一個線程在任何時間改變掉。


            explicit關鍵字


            我們在編寫應用程序的時候explicit關鍵字基本上是很少使用,它的作用是"禁止單參數構造函數"被用于自動型別轉換,
            其中比較典型的例子就是容器類型,在這種類型的構造函數中你可以將初始長度作為參數傳遞給構造函數.
            例如:
            你可以聲明這樣一個構造函數
            class Array
            {
            public:
            ?explicit Array(int size);
            ?......
            };
            在這里explicit關鍵字起著至關重要的作用,如果沒有這個關鍵字的話,這個構造函數有能力將int轉換成Array.一旦這種
            情況發生,你可以給Array支派一個整數值而不會引起任何的問題,比如:
            Array arr;
            ...
            arr = 40;
            此時,C++的自動型別轉換會把40轉換成擁有40個元素的Array,并且指派給arr變量,這個結果根本就不是我們想要的結果.如果
            我們將構造函數聲明為explicit,上面的賦值操作就會導致編譯器報錯,使我們可以及時發現錯誤.
            需要注意的是:explicit同樣也能阻止"以賦值語法進行帶有轉型操作的初始化";
            例如:
            Array arr(40);//正確
            Array arr = 40;//錯誤

            看一下以下兩種操作:
            X x;
            Y y(x);//顯式類型轉換
            另一種
            X x;
            Y y = x;//隱式類型轉換

            這兩種操作存在一個小小的差別,第一種方式式通過顯式類型轉換,根據型別x產生了型別Y的新對象;第二種方式通過隱式轉換
            產生了一個型別Y的新對象.
            explicit關鍵字的應用主要就是上面所說的構造函數定義種,參考該關鍵字的應用可以看看STL源代碼,其中大量使用了該關鍵字

            ?

            __based關鍵字


            該關鍵字主要用來解決一些和共享內存有關的問題,它允許指針被定義為從某一點開始算的32位偏移值,而不是內存種的絕對位置
            舉個例子:

            typedef struct tagDEMOSTRUCT {
            ?int a;
            ?char sz[10];
            } DEMOSTRUCT, * PDEMOSTRUCT;

            HANDLE hFileMapping = CreateFileMapping(...);
            LPVOID lpShare = (LPDWORD)MapViewOfFile(...);

            DEMOSTRUCT __based(lpShare)* lpDemo;

            上面的例子聲明了一個指針lpDemo,內部儲存的是從lpShare開始的偏移值,也就是lpHead是以lpShare為基準的偏移值.
            上面的例子種的DEMOSTRUCT只是隨便定義的一個結構,用來代表任意的結構.

            雖然__based指針使用起來非常容易,但是,你必須在效率上付出一定的代價.每當你用__based指針處理數據,CPU都必須
            為它加上基地址,才能指向真正的位置.

            在這里我只是介紹了幾個并不時很常見的關鍵字的意義即用法,其他那些常見的關鍵字介紹他們的文章已經不少了在這里
            就不再一一介紹了.希望這些內容能對大家有一定的幫助!

            posted on 2006-10-21 13:20 Jerry Cat 閱讀(2349) 評論(7)  編輯 收藏 引用

            FeedBack:
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-21 13:23 | Jerry Cat
            ----- 大伙兒幫下忙咯 ----- 不知here有沒有南京的網友, 我以前南京的一個同學趙小姐要我幫忙發個租房消息, 某雖不懂但老同學的事也不好推, 將就如下:

            現有位于南京市江寧區"武夷花園"小區的三室一廳住房一套, 水,電,氣齊全, 室內有熱水器, 電視機及床, 衣柜等簡單家具. 整租, 與人合租均可. 聯系人: 趙小姐 電話:025-83909202  回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-21 14:29 | ^_^
            似乎是轉貼啊,很久以前就看過了。
            __based標準里面真的有這個關鍵字的存在嗎?這俺到不知道了,水平有限。
            有一個模板相關的關鍵字export沒提出來,雖然現在的編譯器決大部分都不支持(有支持的俺就不知道了)。  回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-23 09:46 | run_mei
            mutable不知道,說明你不常用const關鍵字
            volatile沒有用的東西,忘記它吧.
            explicit不知道,說明你對c++類型轉換不熟,很危險.
            __based非標準,忘記它吧.  回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-23 09:57 | run_mei
            與volatile類似還有register
              回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-23 19:43 | 空明流轉
            volatile不能算沒用,并發設計的時候,關鍵的地方是一定要加上的,免得編譯器錯誤的優化了代碼。  回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-23 23:07 | Jerry Cat
            volatile很有用的, 只不過國內的教科書老是在次語焉不詳. register在以前的turbo C中倒是用過的, 自從C++后倒無緣相見:)  回復  更多評論
              
            # re: "C++你知道嗎?" 之 幾個冷僻的關鍵字
            2006-10-25 00:31 | 小山日志
            volatile可不能隨便忘記的,并發時代就要到來啦  回復  更多評論
              

            <2006年9月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            1234567

            常用鏈接

            留言簿(7)

            隨筆檔案

            最新隨筆

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            国产成人精品久久| 久久精品国产只有精品2020| 久久国产精品99精品国产987| 久久影视综合亚洲| 国产精品VIDEOSSEX久久发布| 国产精品99久久免费观看| 久久亚洲精品无码aⅴ大香| 久久久不卡国产精品一区二区 | 精品国产乱码久久久久久人妻| 色综合合久久天天综合绕视看| 91精品国产色综合久久| 奇米影视7777久久精品| 国产美女久久精品香蕉69| 7777久久亚洲中文字幕| 久久精品国产秦先生| 99久久久国产精品免费无卡顿| 久久久精品午夜免费不卡| 99久久国产免费福利| 久久亚洲国产午夜精品理论片| 婷婷综合久久狠狠色99h| 久久久久综合网久久| 国内精品久久久久久久久电影网| 久久国产精品一区| yy6080久久| 久久久久久人妻无码| 九九99精品久久久久久| 久久综合色区| 久久夜色精品国产噜噜亚洲AV| 久久er热视频在这里精品| 日韩亚洲国产综合久久久| 99久久无色码中文字幕人妻| 国产美女久久久| 无码国内精品久久综合88| 国产精品久久久久无码av| 久久精品无码一区二区三区日韩| 久久九九兔免费精品6| 久久国产精品99精品国产987| 热久久最新网站获取| 伊人久久免费视频| 狠狠色狠狠色综合久久| 久久精品亚洲福利|