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::**时,使用强制转化时需要使用符号''来强制转化。当然
内存地址转化也同样如此。
 
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在使用gdb调试时,可以使用commands命令来设置断点,并指定在每次到达该断点时要执行的一组命令。该命令的使用方法如下: 1. 首先,使用gdb启动程序,例如:gdb ./gdbdebug 2. 接下来,使用break命令设置一个断点,例如:b abc.cpp:10 3. 然后,使用commands命令将以下命令添加到指定的断点上: commands breakpoint-number 命令1 命令2 ... end 这里的breakpoint-number是断点号,表示将以下命令添加到指定的断点上。可以添加任何有效的GDB命令,每行一个命令,以end结束。 例如,假设我们有以下示例代码: ```cpp using namespace std; void func1() // 行号为10 { cout << "before" <<endl; } int main() { func1(); return 0; } ``` 我们可以使用以下命令来设置断点并指定commands: ```bash gdb a.out b abc.cpp:10 commands 1 silent prints "after" return c end run ``` 这样,在运行程序时,当程序执行到这个断点时,GDB会执行commands指定的命令列表,然后继续执行程序。在这个例子,commands命令会将输出修改为"after",然后继续执行程序,最终输出结果为"after"。 参考资料: 1. [GDB]断点(breakpoint)命令列表:commands、silent 2. 书籍《软件调试的艺术.pdf》2.11节 3. gdb commands命令用法。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [GDG调试技巧之命令列表(commands)](https://blog.csdn.net/warden007/article/details/82888035)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [GDB Commands用法](https://blog.csdn.net/qq_33726635/article/details/117199722)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值