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

相关文章

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

SpringBoot通过main方法启动web项目实践

《SpringBoot通过main方法启动web项目实践》SpringBoot通过SpringApplication.run()启动Web项目,自动推断应用类型,加载初始化器与监听器,配置Spring... 目录1. 启动入口:SpringApplication.run()2. SpringApplicat

PostgreSQL简介及实战应用

《PostgreSQL简介及实战应用》PostgreSQL是一种功能强大的开源关系型数据库管理系统,以其稳定性、高性能、扩展性和复杂查询能力在众多项目中得到广泛应用,本文将从基础概念讲起,逐步深入到高... 目录前言1. PostgreSQL基础1.1 PostgreSQL简介1.2 基础语法1.3 数据库

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定

Python文本相似度计算的方法大全

《Python文本相似度计算的方法大全》文本相似度是指两个文本在内容、结构或语义上的相近程度,通常用0到1之间的数值表示,0表示完全不同,1表示完全相同,本文将深入解析多种文本相似度计算方法,帮助您选... 目录前言什么是文本相似度?1. Levenshtein 距离(编辑距离)核心公式实现示例2. Jac

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

SQL Server 查询数据库及数据文件大小的方法

《SQLServer查询数据库及数据文件大小的方法》文章介绍了查询数据库大小的SQL方法及存储过程实现,涵盖当前数据库、所有数据库的总大小及文件明细,本文结合实例代码给大家介绍的非常详细,感兴趣的... 目录1. 直接使用SQL1.1 查询当前数据库大小1.2 查询所有数据库的大小1.3 查询每个数据库的详