C/C++程序内存地址出错查找方法简介

2024-01-05 09:32

本文主要是介绍C/C++程序内存地址出错查找方法简介,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

很多时候,我们写的程序都会出现程序的非法推出,如果程序很大,此类错误非常难查,下面介绍两种方法,从错误的内存地址反向查找出问题的程序:

<1>.通过汇编去查找.
linux 平台:
  1. 在程序信号处理部分, 加入代码捕捉引起错误点的地址,简单来说,方法就是在注册自己的信号处理函数,在这个函数中加入获取内存错误地址的代码,并把结果写到一个日志文件中。
  2. 编译 DEBUG 版本 程序 (compile 时用 -g , 生成可执行文件后不用 strip 去掉symbol 信息)
  3. 在程序出问题时, 查看日志记录, 得到错误点的地址.
  4. 用objdump -S 导出Debug 版本的汇编代码, 查找错误地址, 则得出那条语句出错.
windows 下c 语言调试
  1. release 版编译/连接选项, 把"generate debug info/" 打钩选择
  2.dumpbin /DISASM /OUT:dump.out.txt.1 prep.exe 可反编译exe文件
  3.得到程序非法地址(可从管理工具-》事件查看器里得到),与汇编比较。

 

<2>.我个人比较常用的.主要是WINDOWS下的.以VC2003为例.

    1.打开工程的属性,在配置属性里面,选择链接器/调试.在右边的生成映射文件选择:是(/MAP)
映射文件名:$(IntDir)/XXX.map.最后映射行选择:是(/MAPINFO:LINES).这样RELEASE编译的
时候就会产生程序的MAP文件,在这个文件里面记录了程序各接口,函数的调用地址等.而最后
选择的映射行则是对每个CPP文件都产生.cod文件,这里面记录了此CPP文件在内存中的地址
映射.我们就是通过MAP文件和这些COD文件查找具体的出错位置.
2.设置完了以上设置后,最好在程序里面写一个地址出错的LOG文件,方便程序出错的时候,可以
查找出正确的位置.下面来说说具体的查找过程:
如 你在执行一个程序的时候,报了内存错误,错误地址为:001B:00474A66.后面的00474A66就是
它的出错地址.然后我们先打开该程序的MAP文件(一个程序一般只有一个MAP文件),接着在MAP
文件中查找与00474A66最接近的地址是哪个,一定要找比它小一点的地址.因为MAP文件中记录的
都是程序各函数的入口地址,所以这个00474A66一定是在某个地址与另一个地址之间.找到那个比
出错地址小一点的入口地址后.在那个入口地址的那一行,你可以看到如:
?CheckCharName@CUnitSvrModule@@IAEHPBD@Z 004748e0 f i XXXX.obj.
?CheckCharName这是函数名.CUnitSvrModule这是类名.004758e0 这是入口地址.
这个XXXX.obj就是出错的CPP文件,你再打开XXXX.COD这个文件,通过函数名去找到此函数的入口位置.
如:
;COMDAT ?CheckCharName@CUnitSvrModule@@IAEHPBD@Z
_TEXT SEGMENT
__$EHRec$ = -12      ; size = 12
_szNick$ = 8      ; size = 4
_szNickName$ = 8     ; size = 4
?CheckCharName@CUnitSvrModule@@IAEHPBD@Z PROC NEAR ; CUnitSvrModule::CheckCharName, COMDAT
; _this$ = ecx
如果看到这种形式.说明已经找到函数的代码执行处了,下面的内容就是此函数的汇编代码.你再根据
上面的地址差找到具体在哪一行就行了. 地址差:= 00474A66(出错地址)-004748e0(入口地址).
3.大致的查找过程就是这样.上面也已经说过了,在程序里面写一个内存出错的LOG文件很重要,这样方便
程序员在事后维护的时候有据可查.而且编译之前一定要先设置映射文件及映射行.

这篇关于C/C++程序内存地址出错查找方法简介的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/572417

相关文章

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Redis实现延迟任务的三种方法详解

《Redis实现延迟任务的三种方法详解》延迟任务(DelayedTask)是指在未来的某个时间点,执行相应的任务,本文为大家整理了三种常见的实现方法,感兴趣的小伙伴可以参考一下... 目录1.前言2.Redis如何实现延迟任务3.代码实现3.1. 过期键通知事件实现3.2. 使用ZSet实现延迟任务3.3

idea maven编译报错Java heap space的解决方法

《ideamaven编译报错Javaheapspace的解决方法》这篇文章主要为大家详细介绍了ideamaven编译报错Javaheapspace的相关解决方法,文中的示例代码讲解详细,感兴趣的... 目录1.增加 Maven 编译的堆内存2. 增加 IntelliJ IDEA 的堆内存3. 优化 Mave

Java String字符串的常用使用方法

《JavaString字符串的常用使用方法》String是JDK提供的一个类,是引用类型,并不是基本的数据类型,String用于字符串操作,在之前学习c语言的时候,对于一些字符串,会初始化字符数组表... 目录一、什么是String二、如何定义一个String1. 用双引号定义2. 通过构造函数定义三、St

Spring Security方法级安全控制@PreAuthorize注解的灵活运用小结

《SpringSecurity方法级安全控制@PreAuthorize注解的灵活运用小结》本文将带着大家讲解@PreAuthorize注解的核心原理、SpEL表达式机制,并通过的示例代码演示如... 目录1. 前言2. @PreAuthorize 注解简介3. @PreAuthorize 核心原理解析拦截与

一文详解JavaScript中的fetch方法

《一文详解JavaScript中的fetch方法》fetch函数是一个用于在JavaScript中执行HTTP请求的现代API,它提供了一种更简洁、更强大的方式来处理网络请求,:本文主要介绍Jav... 目录前言什么是 fetch 方法基本语法简单的 GET 请求示例代码解释发送 POST 请求示例代码解释

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

Feign Client超时时间设置不生效的解决方法

《FeignClient超时时间设置不生效的解决方法》这篇文章主要为大家详细介绍了FeignClient超时时间设置不生效的原因与解决方法,具有一定的的参考价值,希望对大家有一定的帮助... 在使用Feign Client时,可以通过两种方式来设置超时时间:1.针对整个Feign Client设置超时时间

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(