VC项目调试基础


原文链接:http://blog.csdn.net/phunxm/article/details/5203931


一.Debug版本和Release版本的区别

Debug通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序。Release称为发布版本,它往往是进行了各种优化,使得程序在代码大小和运行速度上都是最优的,以便用户很好地使用。

只有Debug版本的程序才能设置断点、单步执行、使用TRACE/ASSERT等调试输出语句。Release版本不包含任何调试信息,所以体积小、运行速度快。

一般而言Debug版本会比Release版本多出*.ilk文件和*.pdb文件。

1.*.ilk文件

ilk后缀全称为“Incremental Linking”,意即增量链接。

VC6中,“Project Setting->Link(Category:General)”默认勾选“Link incrementally”;相应VC2005中,“项目属性->配置属性->链接器->常规->启用增量链接”默认选项为“是(/INCREMENTAL)”。

当选定渐增型编译链接时,链接器自动生成ILK文件,记录链接信息,也就是每次重新编译并不编译所有的源文件,只编译改动过的文件。而编译器怎么知道哪些编译过哪些未编译过呢,除了检查修改时间外,这个ilk文件也是很重要的信息。

在VS2005中,断点变成了感叹号圆圈,若提示“当前不会命中断点。此位置当前尚未加载可执行代码”,则可能是没有打开生成调试信息的/Zi开关。在“项目属性->配置属性->C/C++->常规->调试信息格式”处选择“程序数据库(/Zi)”。在“项目属性->配置属性->链接器->调试->生成调试信息” 处选择“是(/DEBUG) ”。若提示“当前不会命中断点,源代码与原始版本不同。”,尝试将“工具->选项->调试”中“要求源文件与原始版本完成匹配”前的勾去掉,删除Debug目录重新生成解决方案。

/C7

C 7.0- Compatible

目标文件或者可执行文件中包含行号和所有符号调试信息,包括变量名及类型,函数及原型等

/Zi

Program Database

创建一个程序库(PDB),包括类型信息和符号调试信息。

某个源文件中设置的断点老是无效,提示“当前不会命中断点。源代码与原始版本不同”信息,清理项目、重编项目、重新拷贝原文件都解决不了问题。“工具->选项->调试”去掉“要求源文件与原始版本完成匹配”前面的勾。

参考: 《VS2005 当前不会命中断点,还没有为该文档加载任何符号

2.*.pdb文件

pdb后缀全称为“Program Debug Database”,意即程序数据库文件。

VC6中,“Project Setting->Link (Category:General)”默认勾选“Generate debug info”,“Project Setting->Link (Category:Customize)”默认勾选“Use program database”;相应VC2005中,“项目属性->配置属性->链接器->调试”,默认“生成调试信息”选项为“是(/DEBUG)”,默认“生成程序数据库文件”处填写“.\Debug\*.pdb”。该选项对应的编译开关为/PDB。

符号文件(Symbol Files)是一个数据信息文件,它包含了应用程序二进制文件(比如EXE、DLL等)的调试信息,专门用来作调试之用,最终生成的可执行文件在运行时并不需要这个符号文件,但你的程序中所有的变量信息都记录在这个文件中。所以调试应用程序时,这个文件是非常重要的。用VC和 WinDbg调试程序时都要用到这个文件。

关于符号文件,参考《PDB Files(C++)符号文件—Windows 应用程序调试必备》。

二.VC常用调试操作

快捷键

作用

F5

开始逐断点调试

Shift+F5

停止调试

F9

当前行增加/删除断点

Ctrl+F9

当前行启用/禁用断点

Ctrl+B

增加函数断点

Alt+F9

查看所有断点(Debug Window)

Ctrl+Shift+F9

删除所有断点

F10

单步调试,遇到子函数跳过Step Over

F11

单步调试,遇到子函数进入Step Into

Shift+F11

从当前函数跳出返回上一级callback

一般F9设断点,F5启动后,F10、F11和Shift+F11交替使用,以在多个断点处调试。

Debug菜单中可以“启用所有断点”、“禁用所有断点”;右键“breakpoints”可以设置“Condition”、“Filter”和“Hit Count”等条件断点


三.源码跟踪调试

我们在VC2005中printf函数调用处设置断点,F11 将打开printf函数的源码文件C:\ProgramFiles\Microsoft Visual Studio 8\VC\crt\src\printf.c进入printf函数内部,实际上“工具->选项->项目和解决方案->VC++目录->源文件”中包含了crt源代码的路径:$(VCInstallDir)crt\src。

而在VC6中,C:\ProgramFiles\Microsoft Visual Studio\VC98\CRT\SRC不存在,故试图F11进入printf内部时,将弹出“Find Source”对话框,提示“Please enter the path for PRINTF.C”文件。

调试VC时,编译器之所以能够跟踪进入C库函数,是因为编译器本身附带提供了Debug版本的CRT库。

VC6中的MSVCRTD.DLL-MSVCRTD.LIB-MSVCRTD.PDB(对应VC2005中msvcr80d.dll-msvcrtd.lib)为C Runtime Library的Debug版本,其中包含了C运行库的调试信息。

VC6中的MSVCP60D.DLL-MSVCPRTD.LIB-MSVCP60D.PDB(对应VC2005中msvcp80d.dll- msvcprtd.lib)为C++Runtime Library的Debug版本,其中包含了C++运行库的调试信息。

鉴于Debug版本对调试信息的记录和Dll->Lib(*.obj,*.lib)->Src(*.c,*.cpp)的对应关系,在VC中我们就可以设置断点跟踪调试代码。

 

四.反汇编

在VC2005中,“文件属性->C/C++->输出文件->汇编输出”默认为“无列表”,可选项如下:

(1)“仅列出程序集(/FA)”:生成FA.asm,有行号无源代码。

(2)“带源代码的程序集(/FAs)”:生成FAs.asm,有行号和源代码。

(3)“带机器码的程序集(/FAc)”:生成FAc.cod,在FA.asm基础上列出了机器码。

(4) “程序集、机器码和源代码(/FAcs)”:生成FAcs.cod,在FAc.cod的基础上列出了对应的源代码。

在Debug状态下,“调试->窗口->反汇编(Alt+8)”,其中“Alt+8” 反汇编显示的即为/FAs结果。在调试过程中,可“Alt+5”呼出寄存器窗口,查看数据寄存器(EAX/EBX/ECX/EDX)、变址寄存器(ESI/EDI)、控制寄存器(EFL/EIP)和指针寄存器(ESP/EBP)的实时状态值。

可在命令窗口中输入 “cl /?”查看帮助中的“-OUTPUT FILES-”部分的说明。详情可参考MSDN中的说明:《/FA, /Fa (ListingFile)》《/FA、/Fa(列表文件)》。

 

五.链接库(lib、dll)项目和应用程序(exe)项目联合调试

对于链接库项目,必须要导入一个能够调用它的测试程序进来,通过应用程序的执行来达到调试的目的。

既然是调试,当然静态或动态链接库项目应生成带调试信息的Debug版本,VC6“Project Setting->Link (Category:General)”默认勾选“Generate debug info”,相应VC2005中的“项目属性->配置属性->链接器->调试->生成调试信息”默认选项为“是(/DEBUG)”。

1.*.obj文件和*.lib文件

obj:Intermediate file,CPP对应的二进制代码格式,是未经重定位的!

lib:Object File Library,若干个obj的集合,本质与obj相同!

使用时,obj和lib是没有本质区别的,lib就是obj,使用lib地方都可以使用obj。

对于*.cpp文件中的static变量或函数,在*.obj或*.lib中既有符号信息又有实体定义。

对于*.cpp文件中的extern变量或函数,在*.obj或*.lib中一般都会有符号信息,实体既可以定义在本文件中,也可以在其他文件中定义。对于extern变量或函数,链接时,将在所有的*.obj或*.lib中查找实体。

2.LibExe协调

新建解决方案D:\ LibDemo\LibDemo(.dsw,.sln),在其下新建静态库(Win32 Static Library)项目MyLib.dsp,.vcproj)和静态库测试项目(Win32 ConsoleApplicationMyLibDemo.dsp,.vcproj)。首先需要为MyLibDemo项目附加头文件包含和链接库包含。

(1)若设置MyLibDemo为启动项目,执行F5调试命令,则在MyLibMyLibDemo中的断点处都会暂停。在MyLibDemo中可F11进入MyLib中的函数定义处。

(2)若设置MyLib为启动项目,则要配置外部exe调用程序:

VC6中,“Project Setting->Debug(Category:General)”中的“Executable for debugsession”处填写exe路径,例如:D:\LibDemo\MyLibDemo\Debug\MyLibDemo.exe。相应VC2005中,“项目属性->配置属性->调试->命令”处填写exe路径。

执行F5调试命令,则在MyLibMyLibDemo中的断点处都会暂停。效果同(1)。

对于不同解决方案(.dsw,.sln)下的Lib项目和Exe项目协同调试方法同上。

3.DllExe协调

dll项目往往会附带生成一个对应的lib,是dll导出函数和符号链接信息。同lib项目生成的lib有点不同,这里只有原型信息,实体代码在dll中。dll是可实际运行的二进制代码,其中有定位代码。

新建解决方案D:\DllDemo\DllDemo(.dsw,.sln),在其下新建动态库(Win32 Dynamic-Link Library)项目MyDll.dsp,.vcproj)和动态库测试项目(Win32 Console ApplicationMyDllDemo.dsp,.vcproj)。

同Lib和Exe协调一样,首先需要为Exe项目附加头文件(.h)包含和链接库(.lib)包含。

(1)若设置MyDllDemo为启动项目,为了使MyDll.dll对调试exe可见,设置MyDllDemo的调试工作目录(Working directory)为..\MyDll\Debug

执行F5调试命令,则在MyDllMyDllDemo中的断点处都会暂停。在MyDllDemo中可F11进入MyDll中的函数定义处。

(2)若设置MyDll为启动项目(例如def显式动态链接库的调试),则要配置外部exe调用程序:

VC6中,“Project Setting->Debug(Category:General)”中的“Executable for debugsession”处填写exe路径,例如:D:\DllDemo\MyDllDemo\Debug\MyDllDemo.exe。相应VC2005中,“项目属性->配置属性->调试->命令”处填写exe路径。

为了使MyDll.dll对调试exe可见,设置调试工作目录(Working directory)为.\Debug

执行F5调试命令,则在MyDllMyDllDemo中的断点处都会暂停。效果同(1)。

对于不同解决方案(.dsw,.sln)下的Dll项目和Exe项目协同调试方法同上。

4.生成后事件处理

在(1)和(2)中,除了指定调试工作目录外,还可以处理生成后事件,将生成的链接库文件直接复制到应用程序Debug目录下。具体的,在VC6“Project Setting ->Post-builtstep”或VC2005中“项目属性->配置属性->生成事件->生成后事件”输入生成后处理命令:copy .\Debug\MyDll.dll..\MyDllDemo\Debug\MyDll.dll(有时需要使用xcopy以自动生成目录)。当然,还可以将MyLibMyDll)的输出目录直接指向MyLibDemo(MyDllDemo)Debug下。

 

参考:

    《VisualC++ 6.0调试功能 图解教程



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值