gdb
中namespace的问题
由于
namespace能防止命名冲突,因此在许多工程中会使用namespace。但是gdb
调试的时候如果需要强制将
void*指针转化成namespace中的某个struct或者class时,
会出现一些问题。下面通过举例介绍解决的方法。
测试代码如下所示,总共两个文件,在
test.h中使用了namespace,并在其中定
义了一个
struct(class跟struct类似)。在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::**时,使用强制转化时需要使用符号''来强制转化。当然
内存地址转化也同样如此。