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

            醬壇子

            專注C++技術 在這里寫下自己的學習心得 感悟 和大家討論 共同進步(歡迎批評!!!)

              C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
              66 Posts :: 16 Stories :: 236 Comments :: 0 Trackbacks

            公告

            王一偉 湖南商學院畢業 電子信息工程專業

            常用鏈接

            留言簿(19)

            我參與的團隊

            搜索

            •  

            積分與排名

            • 積分 - 387045
            • 排名 - 64

            最新隨筆

            最新評論

            閱讀排行榜

            評論排行榜

            關于gcnew?? (轉)


            C++/CLI中使用gcnew關鍵字表示在托管堆上分配內存,并且為了與以前的指針區分,用^來替換* ,就語義上來說他們的區別大致如下:

              1.???? gcnew返回的是一個句柄(Handle),而new返回的是實際的內存地址.
              2.???? gcnew創建的對象由虛擬機托管,而new創建的對象必須自己來管理和釋放.
            ?
              當然,從程序員的角度來說,管它是句柄還是什么其他的東西,總跑不掉是對某塊內存地址的引用,實際上我們都可以理解成指針.下面我們就寫一段代碼來測試一下好了.
            ?
            using namespace System;
            ?
            ref class Foo
            {
            public:
            ??? Foo()
            ??? {
            ?????? System::Console::WriteLine("Foo::Foo");
            ??? }
            ??? ~Foo()
            ??? {
            ?????? System::Console::WriteLine("Foo::~Foo");
            ??? }
            public:
            ??? int m_iValue;
            };
            ?
            int _tmain()
            {
            ??? int* pInt = new int;
            ??? int^ rInt = gcnew int;
            ??? Foo^ rFoo = gcnew Foo;
            ?
            ??? delete rFoo;
            ??? delete rInt;
            ??? delete pInt;
            }
            ?
              我把調試的時候JIT編譯的匯編代碼擇錄了部分如下顯示(請注意紅色部分):

            ??? int* pInt = new int;
            0000004c? mov???????? ecx,4
            00000051? call??????? dword ptr ds:[03B51554h]
            00000057? mov???????? esi,eax
            00000059? mov???????? dword ptr [esp+18h],esi
            ??? int^ rInt = gcnew int;
            0000005d? mov???????? ecx,788EF9D8h
            00000062? call??????? FCFAF66C
            00000067? mov???????? esi,eax
            00000069? mov???????? dword ptr [esi+4],0
            00000070? mov???????? edi,esi
            ??? Foo^ rFoo = gcnew Foo;
            00000072? mov???????? ecx,3B51768h
            00000077? call??????? FCFAF66C
            0000007c? mov???????? esi,eax
            0000007e? mov???????? ecx,esi
            00000080? call??????? dword ptr ds:[03B517ACh]
            00000086? mov???????? dword ptr [esp+1Ch],esi
            ?
            ??? delete rFoo;
            0000008a? mov???????? ebx,dword ptr [esp+1Ch]
            0000008e? test??????? ebx,ebx
            00000090? je????????? 000000A4
            00000092? mov???????? ecx,ebx
            00000094? call??????? dword ptr ds:[03FD0028h]
            0000009a? mov???????? dword ptr [esp+14h],0
            000000a2? jmp???????? 000000AC
            000000a4? mov???????? dword ptr [esp+14h],0
            ??? delete rInt;
            000000ac? mov???????? edx,edi
            000000ae? mov???????? ecx,788F747Ch
            000000b3? call??????? FC8D20FD
            000000b8? mov???????? ebp,eax
            000000ba? test??????? ebp,ebp
            000000bc? je????????? 000000D0
            000000be? mov???????? ecx,ebp
            000000c0? call??????? dword ptr ds:[03FD0020h]
            000000c6? mov???????? dword ptr [esp+10h],0
            000000ce? jmp???????? 000000D8
            000000d0? mov???????? dword ptr [esp+10h],0
            ??? delete pInt;
            000000d8? mov???????? ecx,dword ptr [esp+18h]
            000000dc? call??????? dword ptr ds:[03B51540h]
            ?
            ?
            ?  我們先看分配內存這部分的代碼
            ?
              1.調用new方式分配
            int* pInt = new int;
            0000004c? mov???????? ecx,4
            00000051? call??????? dword ptr ds:[03B51554h]

              可以看到,和以前在vc6中一樣,分配內存的步驟如下:
              1.? 首先把sizeof(int) = 4 放到ecx中
              2.? 調用operator new 去分配4個字節
              3.? 調用構造函數等等......(這里不是我們的重點)

              成功分配后,會把返回地址放在eax中。
            ?
              2.調用gcnew方式分配
            ??? int^ rInt = gcnew int;
            0000005d? mov???????? ecx,788EF9D8h
            00000062? call??????? FCFAF66C
            。。。
            ??? Foo^ rFoo = gcnew Foo;
            00000072? mov???????? ecx,3B51768h
            00000077? call??????? FCFAF66C

              可以看到gcnew也是通過把一個參數放到ecx中,然后再調用一個函數來完成分配的操作,顯然0x788EF9D8應該是一個地址,而不可能是一個數值。我們可以看到這里gcnew創建兩個不同類型的變量,調用的函數地址卻都是0xFCFAF66C,而存放到ecx中的兩個地址就不一樣。究竟這幾個地址代表什么呢?
            ?
              和new一樣gcnew也是把返回地址放在eax中。我們直接從內存窗口看eax指向的內存塊好了。Aha,看到了沒有?

              這次的eax = 0x00F73404? 對應的內存塊為
            ?
            0x00F73404? d8 f9 8e 78 00 00 00 00 。。。
            ?
              這個不就是 mov 到 ecx中的值么?再回憶昨天寫的分析Object對象布局的文章,可以肯定這個就是 MethodTable地址了,對于這個int來說,后面的4個字節對應的就是存放它的RawData,比如如果你初始化為 4 那么內存對應的就變化為 d8 f9 8e 79 04 00 00 00
            ?
              分析清楚存放到ecx中的是 MethodTable指針,我們再分析那個對應的call函數,從vm的代碼可以看出,有三個全局函數用來根據MethodTable創建對象,同時MethodTable本身也提供一個成員函數Allocate(),只不過這個成員函數也是調用的下面的函數:

            OBJECTREF AllocateObject( MethodTable *pMT )
            OBJECTREF AllocateObjectSpecial( MethodTable *pMT )
            OBJECTREF FastAllocateObject( MethodTable *pMT )
            ?
              其中AllocateObject又是調用AllocateObjectSpecial來完成工作。那么我們調用的應該就是AllocateObject或者FastAllocateObject了。

              在我們的例子里面兩個call的地址都一樣,但是你如果寫下代碼 double ^ pDouble = gcnew double;這個時候的地址是多少?它和int 的一樣么?

              目前我還沒有仔細去研究這個地址到底對應的是該類型的MethodTable::Allocate()或是上面的這三個全局函數,如果對應MethodTable::Allocate(),那么2.0中應該有個MethodTable::FastAllocate()吧,否則應該就是對應的全局函數AllocateObject 以及FastAllocateObject了。過幾天一定要抽空再好好研究一下。
            ?
              下面看對應的delete函數。
            ??? delete pInt;
            000000d8? mov???????? ecx,dword ptr [esp+18h]
            000000dc? call??????? dword ptr ds:[03B51540h]
            ?
            比較簡單,就是傳入地址,然后調用operator delete來釋放類存,會調用析構函數
            ?
              對應的,釋放gcnew創建的對象的代碼如下:
            ??? delete rInt;
            000000ac? mov???????? edx,edi
            000000ae? mov???????? ecx,788F747Ch
            000000b3? call??????? FC8D20FD

              這個也相對簡單,它對應vm里面的一個函數:
            void? CallFinalizer(Thread* FinalizerThread, Object* fobj)

              那么也就是
            fobjà edx
            FinalizerThread à ecx
            Call CallFinalizer
            ?
              但是,請注意!!!!!!!一個類包含析構函數和不包含析構函數,它對應的delete代碼是不一樣的,這點可以通過匯編代碼比較得到,我這里就不多說了。
            posted on 2007-03-20 12:52 @王一偉 閱讀(2977) 評論(0)  編輯 收藏 引用
            亚洲伊人久久成综合人影院 | 青青草原综合久久大伊人| 欧美久久久久久| 国产精品成人精品久久久| 久久久久人妻精品一区| 亚洲第一永久AV网站久久精品男人的天堂AV| 久久久亚洲裙底偷窥综合| 久久亚洲国产中v天仙www| 色婷婷综合久久久久中文一区二区 | 麻豆亚洲AV永久无码精品久久| 午夜精品久久久久成人| 久久婷婷五月综合色高清| 情人伊人久久综合亚洲| 99精品久久久久久久婷婷| 免费一级做a爰片久久毛片潮| 久久93精品国产91久久综合| 91精品国产91久久久久福利| 精品国产VA久久久久久久冰 | 香蕉久久久久久狠狠色| 久久夜色精品国产亚洲| 久久人人爽人人爽人人AV东京热 | 99久久99久久精品国产片果冻| 热久久这里只有精品| 精品无码久久久久久午夜| 久久99热这里只有精品66| 久久99国产精品成人欧美| 日本精品久久久久中文字幕8| 亚洲精品tv久久久久久久久| 一本久久a久久精品亚洲| 久久综合九色综合久99| 久久精品国产72国产精福利| 久久综合亚洲色一区二区三区| 久久久久久久久久免免费精品| 曰曰摸天天摸人人看久久久| 亚洲国产成人久久笫一页 | 久久婷婷五月综合成人D啪| 久久这里只有精品视频99| 亚洲精品国产自在久久| 亚洲伊人久久成综合人影院 | 亚洲午夜精品久久久久久人妖| 亚洲成色999久久网站|