• <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>
            在帖子如何判斷一個(gè)C++對(duì)象是否在堆棧上 中, 又有人提出如何判斷一個(gè)C++對(duì)象是否在堆上。

            其實(shí)我們可以參照那個(gè)帖子的方法類似實(shí)現(xiàn),我們知道堆就是Heap,在windows上我們可以通過(guò)GetProcessHeaps來(lái)得到所有的堆句柄,而我們這里只要知道Windows上的Heap Handle,其實(shí)就是堆的起始地址,就可以寫(xiě)如下代碼了。

            #include <iostream>
            #include <windows.h>

            using namespace std;

            BOOL IsObjectOnHeap(LPVOID pObject)
            {
                BOOL bRet(FALSE);
                DWORD dwHeaps = GetProcessHeaps(0, NULL);
                LPHANDLE pHeaps = new HANDLE[dwHeaps];
                if(pHeaps != NULL)
                {
                    MEMORY_BASIC_INFORMATION mi = {0};
                    GetProcessHeaps(dwHeaps, pHeaps);

                    for(INT i=0; i<dwHeaps; ++i)
                    {
                        VirtualQuery((LPVOID)pHeaps[i], &mi, sizeof(mi));
                        
                        if(pObject >= mi.BaseAddress 
                            && (DWORD)pObject < (DWORD)mi.BaseAddress + mi.RegionSize)
                        {
                            bRet = TRUE;
                            break;
                        }
                    }
                }
                delete []pHeaps;

                return bRet;
            }

            int g_value = 10;

            int main(int argc, char* argv[])
            {
                int nStackValue = 1;
                int* pNew = new int(10);
                int* pNewArray = new int[100];

                static int static_value = 0;
                
                cout << "g_value:" << IsObjectOnHeap(&g_value) << endl;  //false
                cout << "nStackValue:" << IsObjectOnHeap(&nStackValue) << endl;   //false
                cout << "static_value:" << IsObjectOnHeap(&static_value) << endl;    //false
                cout << "pNew:" << IsObjectOnHeap(pNew) << endl;        //true
                cout << "pNewArray:" << IsObjectOnHeap(pNewArray) << endl;    //true

                system("pause");
                return 0;
            }

            以上代碼在Windows下測(cè)試通過(guò)(也只能在Windows上跑),如果有不正確的地方,歡迎指正。

            注: 上面關(guān)于對(duì)象是否在堆上的判斷應(yīng)該是不對(duì),因?yàn)槎褍?nèi)存不是連續(xù)內(nèi)存,內(nèi)部是通過(guò)類似鏈表的結(jié)構(gòu)來(lái)實(shí)現(xiàn)的,<<軟件調(diào)試>>里有相關(guān)介紹,也可以通過(guò)WinDbg的 !address 命令查看內(nèi)存分布
            posted on 2012-05-12 14:30 Richard Wei 閱讀(5435) 評(píng)論(8)  編輯 收藏 引用 所屬分類: C++

            FeedBack:
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-12 22:57 | Duwen
            其實(shí)判斷一個(gè)對(duì)象是在堆上還是在棧上根本不必這么復(fù)雜,因?yàn)?系統(tǒng)默認(rèn)棧地址是固定的,棧空間最多長(zhǎng)到1MB,若超過(guò)1MB會(huì)引發(fā)棧耗盡異常,所以只需判斷是否在棧上,就只是看看有沒(méi)有在這個(gè)地址區(qū)間內(nèi)  回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-12 23:05 | Richard Wei
            @Duwen
            堆棧大小一般默認(rèn)是1M,但是VC在編譯時(shí)是可以通過(guò)/STACK :reserve[,commit] 設(shè)置的

            另外對(duì)象所在位置, 除了堆(Heap)和棧(Stack),還有靜態(tài)存儲(chǔ)區(qū)等  回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-13 08:12 | Duwen
            是的,棧基址的卻可以通過(guò)/STACK開(kāi)關(guān)指定.我說(shuō)的方法只是用于一些只有一個(gè)線程簡(jiǎn)單項(xiàng)目的判斷,如果要對(duì)多線程,就得用你這個(gè)方法了,其實(shí)簡(jiǎn)單的判斷單線程程序,我這個(gè)方發(fā)也是可行的,因?yàn)槌绦蛑腥绻亲约憾x的變量只能在三個(gè)地方,全局區(qū),棧區(qū),堆,當(dāng)然手工添加PE節(jié)區(qū)的情況除外,比如創(chuàng)建dll時(shí)加一個(gè)共享段,然后把變量定義在這個(gè)共享段中,全局區(qū)是存放靜態(tài)數(shù)據(jù)和全局變量,所以如果我們要測(cè)試的不是靜態(tài)變量或者全局變量,那么它不在堆區(qū)就在棧.至于靜態(tài)存儲(chǔ)區(qū)是個(gè)廣義的說(shuō)法,可以是全局區(qū),棧,文字常量區(qū),代碼區(qū).

            對(duì)了,為了驗(yàn)證new 時(shí)c/c++ CLR會(huì)創(chuàng)建自己的堆我做了測(cè)試,發(fā)現(xiàn)一個(gè)有趣的事,就是在程序中,一開(kāi)始,就有4個(gè)堆,我把所有標(biāo)準(zhǔn)C++文件全部移出,只留一個(gè)Windows.h, 并連接release版c/c++ CLR,結(jié)果還是4,我把windows.h移出結(jié)果也是四,即使我在我的Cpp中沒(méi)有new,結(jié)果還是4,發(fā)現(xiàn)要想驗(yàn)證這個(gè)還真是不好辦,因?yàn)轭^文件中庫(kù)函數(shù)也可以new,也可以創(chuàng)建堆等等有太多不可控制因素,我能想到的就是看源碼了,但我還真沒(méi)找到源碼,不知你有沒(méi)有什么方法.
              回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-13 08:28 | 13174115
            那是不是要得到當(dāng)前運(yùn)行函數(shù)的地址
            然后去比對(duì)地址?
              回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-13 09:52 | Richard Wei
            @Duwen
            windows NT內(nèi)核初始化時(shí)就有3個(gè)堆,且余留大小都是1MB,提交大小是8KB,其中一個(gè)是系統(tǒng)默認(rèn)堆, 我們調(diào)用GlobalAlloc(),就是在默認(rèn)堆里分配的內(nèi)存。
            C/C++ CRT會(huì)創(chuàng)建自己的堆,我們平時(shí)new對(duì)象,都是在該堆里分配的內(nèi)存。所以我們一般看到的程序起碼有4個(gè)堆。
            另外如果你用了COM組件(調(diào)用了::CoInitialize(NULL)), 他也會(huì)建立自己的2個(gè)堆,CoTaskMemAlloc() 和 SysAllocString()都在在里面分配的內(nèi)存。
            一般寫(xiě)應(yīng)用程序,理解到此就夠了,再深入就要看Windows內(nèi)核的源代碼了。
              回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-13 11:54 | anexx
            @Duwen
            這個(gè)不是windows 頭文件的問(wèn)題,而是c++ 運(yùn)行時(shí)的問(wèn)題。你如果把入口函數(shù)設(shè)置成main或winmain而不用編譯器自動(dòng)生成的,就不一樣了。  回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-05-13 21:33 | Duwen
            @anexx,Richard wei
            我在crt0.c中找到了__tmainCRTStartup的源碼,找到了,
            if ( !_heap_init(1) ) /* initialize heap */
            fast_error_exit(_RT_HEAPINIT); /* write message and die */
            其中_heap_init(1) 里有:
            _crtheap = HeapCreate( mtflag ? 0 : HEAP_NO_SERIALIZE,
            BYTES_PER_PAGE, 0 )) == NULL )
            而_crtheap是個(gè)全局句柄。

            Richard wei說(shuō)win NT 內(nèi)核初始化時(shí)就有3個(gè)堆...,我想多了解點(diǎn),于是今天去圖書(shū)館找了好幾本書(shū),都沒(méi)有找到相關(guān)內(nèi)容,不過(guò)由于時(shí)間關(guān)系,WRK里還沒(méi)找,

              回復(fù)  更多評(píng)論
              
            # re: 如何判斷一個(gè)C++對(duì)象是否在堆上
            2012-08-13 14:30 | Richard Wei
            上面關(guān)于對(duì)象是否在堆上的判斷應(yīng)該是不對(duì),因?yàn)槎褍?nèi)存不是連續(xù)內(nèi)存,內(nèi)部是通過(guò)類似鏈表的結(jié)構(gòu)來(lái)實(shí)現(xiàn)的,<<軟件調(diào)試>>里有相關(guān)介紹,也可以通過(guò)WinDbg的 !address 命令查看內(nèi)存分布  回復(fù)  更多評(píng)論
              
            一级a性色生活片久久无| 午夜精品久久久久久久无码| 久久久久久人妻无码| 亚洲午夜久久久精品影院| 亚洲国产精品综合久久网络 | 久久久精品一区二区三区| 久久国产高清一区二区三区| 日产精品久久久久久久性色| 久久久久国产一区二区| 久久夜色精品国产欧美乱| 国产亚洲色婷婷久久99精品91| 一本一本久久aa综合精品| 久久久WWW成人免费毛片| 999久久久免费精品国产| 亚洲欧美一级久久精品| 久久99精品久久久久久不卡| 国产成人精品免费久久久久| 国内精品伊人久久久影院| 久久97久久97精品免视看秋霞| 精品国产乱码久久久久久1区2区| 人妻无码精品久久亚瑟影视| 久久久久亚洲AV成人网人人软件| 97超级碰碰碰久久久久| 一本色道久久综合亚洲精品| 日日狠狠久久偷偷色综合96蜜桃| 国产福利电影一区二区三区久久老子无码午夜伦不 | 色偷偷888欧美精品久久久| 久久天天躁狠狠躁夜夜躁2014| 精品久久久久久久久久久久久久久| 久久99亚洲网美利坚合众国| 性高湖久久久久久久久| 久久久久亚洲AV无码观看| 女人高潮久久久叫人喷水| 天天做夜夜做久久做狠狠| 久久99精品久久久久久噜噜 | 亚洲国产精品无码久久久蜜芽 | 999久久久免费国产精品播放| 国产精品久久亚洲不卡动漫| 久久美女网站免费| 99久久伊人精品综合观看| 国产2021久久精品|