gdb中namespace的問題

 

gdb中namespace的問題
 
    由於namespace能防止命名衝突,因此在許多工程中會使用namespace。但是gdb
調試的時候如果需要強制將void*指針轉化成namespace中的某個struct或者class時,
會出現一些問題。下面通過舉例介紹解決的方法。
 
    測試代碼如下所示,總共兩個文件,在test.h中使用了namespace,並在其中定
義了一個structclassstruct類似)。在test.cpp中展示了兩種使用namespace
方法,並定義了一個void * 的指針該指針指向namespace結構定義的一個變量,主
要展示如何在gdb中察看該void*指針的內容。主要使用了gcc 3.2.2和gcc 3.4.3兩個
編譯器(分別對應redhat 9.0和redhat AS4)。
 
/****************************
 test.h
*****************************/
ifndef __TEST_H__
#define __TEST_H__
 
namespace MYNAMESPACE
{
        int iTest;
        struct TTest
        {
                int     _iVal;
                char _cVal;
        };
}
 
#endif
 
 
/****************************
 test.cpp
*****************************/
#include <iostream>
#include "test.h"
 
using namespace std;
 
int main()
{
        /* way 1
        using namespace MYNAMESPACE;
        iTest = 10;
        TTest stTest = {10, 'a'};
 
        int * pi = &iTest;
        TTest * pt = &stTest;
        */
 
       ////////////////////////////////////////////
        // way 2
        MYNAMESPACE::iTest = 10;
        MYNAMESPACE::TTest stTest = {10, 'a'};
 
        int * pi = & MYNAMESPACE::iTest;
        MYNAMESPACE::TTest * pt = &stTest;
        ////////////////////////////////////////////
 
        void * pv = &stTest;   //target: get the value of pv in gdb
 
        cout << "pi = " << *pi << endl;
        cout << "pt->_iVal = " << pt->_iVal << endl;
        cout << "pt->_cVal = " << pt->_cVal << endl;
 
        return 0;
}
 
 
上述兩種使用usespace的方式得到的結果均如下:
///////////////////////////////////////////////
//gcc 3.2.2
///////////////////////////////////////////////
(gdb) p pi
$1 = (int *) 0x8049bac
(gdb) p pt
$2 = (TTest *) 0xbffff550
 
///////////////////////////////////////////////
//gcc 3.4.3
///////////////////////////////////////////////
(gdb) p pi
$1 = (int *) 0x8049db8
(gdb) p pt
$2 = (MYNAMESPACE::TTest *) 0xbfeeb840
 
    由上述結果可以看出gcc 3.2.2下識別pt是TTest *類型,因此獲得pv的
值可以強制轉化成TTest*即可,如下所示:
(gdb) p *(TTest*)pv
$3 = {_iVal = 10, _cVal = 97 'a'}
(gdb) p ((TTest*)pv)->_iVal
$4 = 10
(gdb) p ((TTest*)pv)->_cVal
$5 = 97 'a'
 
   但是在gcc 3.4.3中pt卻被識別成MYNAMESPACE::TTest *類型,而像上面
一樣使用強制轉化卻出現問題:
(gdb) p *(MYNAMESPACE::TTest*)pv
A syntax error in expression, near `)pv'.
   因此需要強制表示該符號類型,如下所示:
(gdb) p *('MYNAMESPACE::TTest' *)pv
$3 = {_iVal = 10, _cVal = 97 'a'}
(gdb) p (*('MYNAMESPACE::TTest' *)pv)->_iVal
$4 = 10
(gdb) p (*('MYNAMESPACE::TTest' *)pv)->_cVal
$5 = 97 'a'
   即要使用符號''來強制說明一下,否則gcc會不認識該符號。另外如果在
gcc 3.2.2中使用p *('MYNAMESPACE::TTest' *)pv,則會報錯:
   No symbol "MYNAMESPACE" in current context.
 
   因此,namespace在不同的編譯器中處理是不一樣的,主要還是用p命令來
察看其具體被識別成什麼類型,然後使用強制轉化即可,注意的是如果被識
別成MYNAMESPACE::**時,使用強制轉化時需要使用符號''來強制轉化。當然
內存地址轉化也同樣如此。
 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章