第8部分-2程序调试北京交通大学计算机与信息技术学院教师:林友芳1.概述什么叫调试?程序没有语法错误,但运行结果不符合期望时,采用适当的工具或方法,反复找出程序中存在的问题,进行修改,最终使程序符合要求的过程。调试是一个程序员必须要掌握的基本技巧,对于编写高质量程序非常重要。掌握基本的调试技巧对今后的学习、工作是非常必要的!北京交通大学计算机与信息技术学院教师:林友芳为什么要调试?问题我的程序跟书上一样,怎么就不出结果呢?我的程序没错啊,怎么不出结果啊?我的程序运行结果不正确,如何找到错误?原因程序没有语法错误,不等于就没有语义错误。要确保程序没错,必须对程序进行各种测试2.解决程序中的错误的方法北京交通大学计算机与信息技术学院教师:林友芳基本方法—人工阅读走查,修改程序员一行一行的看程序根据实验数据,在脑子里或在纸上模拟程序运行过程,在脑子里或纸上记录程序中间运行结果。在这过程中发现程序的错误。修改发现错误,重新运行测试,如果还有错,继续查看,修改。这种方法是最为基本、人人都必须掌握的方法。以前程序调试手段以少,人们经常只能用用纸、笔和脑来模拟和记录程序动态运行中的状态。北京交通大学计算机与信息技术学院教师:林友芳基本方法存在的问题这种方法效率不高,属于原始的方法。要求程序员具有很强的源程序理解、运行过程记忆、预测、纠错能力和经验。程序员必须用脑子、纸、笔来模拟程序的运行状态记录程序的执行进度记录所有变量或数据变化过程观察和记住执行过程中环境的变化程序和模块达到一定规模以后,就很难这样方法调试,效率太低,难度太大,很难发现问题,很难记录状态变化过程。程序员脑子中的计算与程序实际完成的计算往往有差异,这样的差异是很难发现的,是很难在纸上体现出来的。北京交通大学计算机与信息技术学院教师:林友芳采用辅助工具来协助调试随着技术的进步,各种IDE的功能越来越强大,人们在IDE中实现并提供许多功能用来辅助程序员对程序进行调试。IDE的辅助功能提供工具让程序员能控制并看到程序的执行进程,如单步执行,执行到某个位置,进入到某个函数内部执行,…在执行过程中,提供工具记录、展示甚至修改环境的状态:变量值,内存数据,函数返回值,程序界面输出结果,…使程序员不需要再在脑子中执行程序使程序员不需要完全只用脑或纸和笔来记录程序的运行状态,并保证中间结果的正确性北京交通大学计算机与信息技术学院教师:林友芳一些值得思考的问题为什么集成开发环境能够让程序员调试程序?是如何做到的?IDE要能使程序员调试一个程序,需要保存关于程序的哪些信息?调试信息3.项目配置与调试ProjectConfigurationanddebugging你是否注意到每个项目中的debug文件夹?注意,本节有许多很理解的概念,如果暂时看不懂,可以先跳过去。北京交通大学计算机与信息技术学院教师:林友芳什么叫项目?项目即project,指通过设计实现一组程序完成用户所需功能的工程及过程。请查看《上机写程序》那个课件关于项目的解释。两个概念Projectsettings—项目设置Projectconfiguration—项目配置北京交通大学计算机与信息技术学院教师:林友芳Projectsettings项目具有许多属性,如:编译结果输出目录工作目录编译参数需要链接的函数库编译完以后做什么操作编译前要做的操作…了解这些属性的意义以后,我们可以对这些属性值进行设置,以影响编译器的编译行为。北京交通大学计算机与信息技术学院教师:林友芳ProjectConfiguration对项目的所有属性的一组设置值统称为一个ProjectConfiguration,即一组ProjectSettings统称为一个ProjectConfiguration.VC提供两组标准项目设置信息的Configuration模板Debug默认的活跃的项目配置模板Release很有经验的程序员也可新建不同于这两组标准模板的项目配置。北京交通大学计算机与信息技术学院教师:林友芳Setactiveprojectconfiguration在同一时刻,只能有一个configuration是有效的或活跃的。在Build菜单下选择Setactiveprojectconfiguration.北京交通大学计算机与信息技术学院教师:林友芳对项目配置的理解可以从这个角度来简化理解(不够准确)每一个project都有一个开发状态,称为debug与release,得到的编译结果分别称为Debug版程序还处于调试状态的程序Release版程序最终提交给用户的可以执行的程序问题Debug版与Release版有什么区别?北京交通大学计算机与信息技术学院教师:林友芳区别标准的Debug版的项目配置指示编译器在编译时,使项目或程序中包含有调试信息,对程序不做任何的优化,便于程序员调试程序。标准的Release版的配置指示编译器在编译时对程序进行优化。得到的结果在代码大小和运行速度上都是较优的,程序中也不包含调试信息。问题:什么叫优化?为什么不能优化北京交通大学计算机与信息技术学院教师:林友芳ProjectSettings北京交通大学计算机与信息技术学院教师:林友芳选择C/C++页,Category中选择general,则出现一个DebugInfo(调试信息)下拉列表框,可供选择的调试信息方式包括:一些编译选项及解释命令行Projectsettings说明无None没有调试信息/ZdLineNumbersOnly目标文件或者可执行文件中只包含全局和导出符号以及代码行信息,不包含符号调试信息/Z7C7.0-Compatible目标文件或者可执行文件中包含行号和所有符号调试信息,包括变量名及类型,函数及原型等/ZiProgramDatabase创建一个程序库(PDB),包括类型信息和符号调试信息。/ZIProgramDatabaseforEditandContinue除了前面/Zi的功能外,这个选项允许对代码进行调试过程中的修改和继续执行。这个选项同时使#pragma设置的优化功能无效看不懂可先不看4.怎么调程序?辅助我们调试程序的工具称为IDE的调试器debugger北京交通大学计算机与信息技术学院教师:林友芳如何进行入调试?方法1.菜单:BuildStartDebugGo方法2.直接点击工具条上的Go按钮方法3.按热键F5想让程序停下来,怎么办?北京交通大学计算机与信息技术学院教师:林友芳设置断点断点是最常用的技巧。断点是调试器设置的一个代码位置。当程序运行到断点时,程序中断执行,回到调试器。调试时,只有设置了断点并使程序回到调试器,才能对程序进行在线调试。北京交通大学计算机与信息技术学院教师:林友芳设置断点方法可以通过下述方法设置断点方法1—简单方法把光标移动到需要设置断点的代码行上,然后按F9快捷键,或者点工具条上的小手图标。方法2—功能更强大的一种方法弹出Breakpoints对话框按快捷键CTRL+B或ALT+F9,或者通过菜单Edit/Breakpoints打开。打开后点击Breakat编辑框的右侧的箭头,选择合适的位置信息。一般情况下,直接选择linexxx就足够了,如果想设置不是当前位置的断点,可以选择Advanced,然后填写函数、行号和可执行文件信息。北京交通大学计算机与信息技术学院教师:林友芳本课件中的采用范例及流程说明跟踪执行从程序开始显示菜单执行第3个功能结束运行的全过程课堂演示程序北京交通大学计算机与信息技术学院教师:林友芳设断点断点标志断点标志主函数里只有两条语句,调用菜单函数后返回。北京交通大学计算机与信息技术学院教师:林友芳设置断点对话框北京交通大学计算机与信息技术学院教师:林友芳去掉断点把光标移动到给定断点所在的行,再次按F9就可以取消断点。同前面所述,打开Breakpoints对话框后,也可以按照界面提示去掉断点。北京交通大学计算机与信息技术学院教师:林友芳条件断点(有点难)可以为断点设置一个条件,这样的断点称为条件断点。对于新加的断点,可以单击Conditions(条件)按钮,为断点设置一个表达式。当这个表达式发生改变时,程序就被中断。北京交通大学计算机与信息技术学院教师:林友芳数据断点(有点难)数据断点只能在Breakpoints对话框中设置。选择“Data”页,就显示了设置数据断点的对话框。在编辑框中输入一个表达式,当这个表达式的值发生变化时,数据断点就到达。一般情况下,这个表达式应该由运算符和全局变量构成,例如:在编辑框中输入SelectTime这个全局变量的名字,那么当程序中有SelectTime++时,程序就将停在这个语句处。北京交通大学计算机与信息技术学院教师:林友芳消息断点(有点难)VC也支持对Windows消息进行截获。他有两种方式进行截获:窗口消息处理函数和特定消息中断。5.设了断点以后,如何走程序?北京交通大学计算机与信息技术学院教师:林友芳进入调试:1.点此按钮2.按F53.BuildStartDebugGo北京交通大学计算机与信息技术学院教师:林友芳黄色右箭头表示当前就要执行的代码行当前执行中的上下文(语境,函数)自动给出的当前语境下的变量值、返回值观察窗口Debug工具栏自定义的观察窗口,可以定义多组北京交通大学计算机与信息技术学院教师:林友芳掌握Debug工具条的主要按钮含义Stopdebugging停止调试stepinto单步跟进,如果当前语句有函数调用,则单步进入函数执行,否则单步执行完一条语句。stepover单步执行完当前语句,若当前语句有函数调用,除非被调用函数中有断点,否则不会跟进函数。stepout执行完当前语句所在函数的执行,返回该函数的调用处。Runtocursor执行到当前光标处,注意观察箭头便于理解记忆北京交通大学计算机与信息技术学院教师:林友芳常用进程控制快捷键快捷键说明F5goSHIFT+F5StepoverCTRL+F5ExecuteprogramF7BuildCTRL+F7CompileF10StepoverCTRL+F10RuntocursorF11StepintoSHIFT+F11StepoutCRTL+SHIFT+F5Restart北京交通大学计算机与信息技术学院教师:林友芳点击Stepinto后,进入函数Menu()执行当前就要执行的代码行语境切换成Menu()6.如何了解执行状态?北京交通大学计算机与信息技术学院教师:林友芳查看数值VC支持查看变量、表达式和内存的值。所有这些观察都必须是在断点中断的情况下进行。观看变量的值最简单,当断点到达时,把光标移动到这个变量上,停留一会就可以看到变量的值。北京交通大学计算机与信息技术学院教师:林友芳watchVC提供一种被称为Watch的机制来观看变量和表达式的值。在断点状态下,在变量上单击右键,选择QuickWatch,就弹出一个对话框,显示这个变量的值。北京交通大学计算机与信息技术学院教师:林友芳watch单击Debug工具条上的Watch按钮,就出现一个Watch视图(Watch1,Watch2,Watch3,Watch4),在该视图中输入变量或者表达式,就可以观察变量或者表达式的值。注意:这个表达式不能有副作用,例如++运算符绝对禁止用于这个表达式中,因为这个运算符将修改变量的值,导致软件的逻辑被破坏。北京交通大学计算机与信息技术学院教师:林友芳点击Stepover后,开始执行当前函数代码自动给出当前语境下变量的值当前执行的代码行北京交通大学计算机与信息技术学院教师:林友芳当前程序运行结果?什么也没有,为什么?因为没有输出任何内容北京交通大学计算机与信息技术学院教师:林友芳再次单击Stepover,执行完第一个printf语句printf语句返回值被自动给出,表示输出了38个字符执行结果?北京交通大学计算机与信息技