《C语言课程序调试课件》由会员分享,可在线阅读,更多相关《C语言课程序调试课件(66页珍藏版)》请在金锄头文库上搜索。
1、第8部分-2 程序调试北京交通大学计算机与信息技术学院 教师: 林友芳1. 概述n什么叫调试?n程序没有语法错误,但运行结果不符合期望时,采用适当的工具或方法,反复找出程序中存在的问题,进行修改,最终使程序符合要求的过程。n调试是一个程序员必须要掌握的基本技巧,对于编写高质量程序非常重要。n掌握基本的调试技巧对今后的学习、工作是非常必要的!北京交通大学计算机与信息技术学院 教师: 林友芳为什么要调试?n问题n我的程序跟书上一样,怎么就不出结果呢?n我的程序没错啊,怎么不出结果啊?n我的程序运行结果不正确,如何找到错误?n原因n程序没有语法错误,不等于就没有语义错误。n要确保程序没错,必须对程序
2、进行各种测试2. 解决程序中的错误的方法北京交通大学计算机与信息技术学院 教师: 林友芳基本方法人工阅读走查,修改n程序员一行一行的看程序n根据实验数据,在脑子里或在纸上模拟程序运行过程,在脑子里或纸上记录程序中间运行结果。n在这过程中发现程序的错误。n修改发现错误,重新运行测试,如果还有错,继续查看,修改。这种方法是最为基本、人人都必须掌握的方法。以前程序调试手段以少,人们经常只能用用纸、笔和脑来模拟和记录程序动态运行中的状态。北京交通大学计算机与信息技术学院 教师: 林友芳基本方法存在的问题n这种方法效率不高,属于原始的方法。n要求程序员具有很强的源程序理解、运行过程记忆、预测、纠错能力和
3、经验。n程序员必须用脑子、纸、笔来模拟程序的运行状态n记录程序的执行进度n记录所有变量或数据变化过程n观察和记住执行过程中环境的变化n程序和模块达到一定规模以后,就很难这样方法调试,效率太低,难度太大,很难发现问题,很难记录状态变化过程。n程序员脑子中的计算与程序实际完成的计算往往有差异,这样的差异是很难发现的,是很难在纸上体现出来的。北京交通大学计算机与信息技术学院 教师: 林友芳采用辅助工具来协助调试n随着技术的进步,各种IDE的功能越来越强大,人们在IDE中实现并提供许多功能用来辅助程序员对程序进行调试。nIDE的辅助功能n提供工具让程序员能控制并看到程序的执行进程,如单步执行,执行到某
4、个位置,进入到某个函数内部执行,n在执行过程中,提供工具记录、展示甚至修改环境的状态:变量值,内存数据,函数返回值,程序界面输出结果,使程序员不需要再在脑子中执行程序使程序员不需要完全只用脑或纸和笔来记录程序的运行状态,并保证中间结果的正确性北京交通大学计算机与信息技术学院 教师: 林友芳一些值得思考的问题n为什么集成开发环境能够让程序员调试程序?是如何做到的?nIDE要能使程序员调试一个程序,需要保存关于程序的哪些信息?n调试信息3. 项目配置与调试Project Configuration and debugging你是否注意到每个项目中的debug文件夹?注意,本节有许多很理解的概念,如
5、果暂时看不懂,可以先跳过去。北京交通大学计算机与信息技术学院 教师: 林友芳什么叫项目?n项目即project,指通过设计实现一组程序完成用户所需功能的工程及过程。n请查看上机写程序那个课件关于项目的解释。n两个概念nProject settings项目设置nProject configuration项目配置北京交通大学计算机与信息技术学院 教师: 林友芳Project settingsn项目具有许多属性,如:n编译结果输出目录n工作目录n编译参数n需要链接的函数库n编译完以后做什么操作n编译前要做的操作nn了解这些属性的意义以后,我们可以对这些属性值进行设置,以影响编译器的编译行为。北京交通
6、大学计算机与信息技术学院 教师: 林友芳Project Configurationn对项目的所有属性的一组设置值统称为一个Project Configuration,即一组Project Settings统称为一个Project Configuration.nVC提供两组标准项目设置信息的Configuration模板nDebug默认的活跃的项目配置模板nReleasen很有经验的程序员也可新建不同于这两组标准模板的项目配置。北京交通大学计算机与信息技术学院 教师: 林友芳Set active project configurationn在同一时刻,只能有一个configuration是有效的
7、或活跃的。n在Build菜单下选择Set active project configuration.北京交通大学计算机与信息技术学院 教师: 林友芳对项目配置的理解n可以从这个角度来简化理解(不够准确)n每一个project都有一个开发状态,称为debug与release,得到的编译结果分别称为nDebug版程序还处于调试状态的程序nRelease版程序最终提交给用户的可以执行的程序n问题nDebug版与Release版有什么区别?北京交通大学计算机与信息技术学院 教师: 林友芳区别n标准的Debug版的项目配置指示编译器在编译时,使项目或程序中包含有调试信息,对程序不做任何的优化,便于程序员
8、调试程序。n标准的Release版的配置指示编译器在编译时对程序进行优化。得到的结果在代码大小和运行速度上都是较优的,程序中也不包含调试信息。问题:什么叫优化?为什么不能优化北京交通大学计算机与信息技术学院 教师: 林友芳Project Settings北京交通大学计算机与信息技术学院 教师: 林友芳选择C/C+页,Category中选择general ,则出现一个Debug Info(调试信息)下拉列表框,可供选择的调试信息 方式包括:一些编译选项及解释命令行命令行Project settings说明说明无None没有调试信息/ZdLine Numbers Only目标文件或者可执行文件中只
9、包含全局和导出符号以及代码行信息,不包含符号调试信息/Z7C 7.0- Compatible目标文件或者可执行文件中包含行号和所有符号调试信息,包括变量名及类型,函数及原型等/ZiProgram Database创建一个程序库(PDB),包括类型信息和符号调试信息。/ZIProgram Database for Edit and Continue除了前面/Zi的功能外,这个选项允许对代码进行调试过程中的修改和继续执行。这个选项同时使#pragma设置的优化功能无效看不懂可先不看4. 怎么调程序?辅助我们调试程序的工具称为IDE的调试器debugger北京交通大学计算机与信息技术学院 教师: 林
10、友芳如何进行入调试?方法1. 菜单:BuildStart DebugGo方法2. 直接点击工具条上的Go按钮方法3. 按热键F5想让程序停下来,怎么办?北京交通大学计算机与信息技术学院 教师: 林友芳设置断点n断点是 最常用的技巧。n断点是调试器设置的一个代码位置。当程序运行到断点时,程序中断执行,回到调试器。n调试时,只有设置了断点并使程序回到调试器,才能对程序进行在线调试。 北京交通大学计算机与信息技术学院 教师: 林友芳设置断点方法n可以通过下述方法设置断点n方法1简单方法n把光标移动到需要设置断点的代码行上,然后按F9快捷键,或者点工具条上的小手图标。 n方法2功能更强大的一种方法n弹
11、出Breakpoints对话框n按快捷键CTRL+B或ALT+F9,或者通过菜单Edit/Breakpoints打开。n打开后点击Break at编辑框的右侧的箭头,选择 合适的位置信息。一般情况下,直接选择line xxx就足够了,如果想设置不是当前位置的断点,可以选择Advanced,然后填写函数、行号和可执行文件信息。 北京交通大学计算机与信息技术学院 教师: 林友芳本课件中的采用范例及流程说明跟踪执行从程序开始显示菜单执行第3个功能结束运行的全过程课堂演示程序北京交通大学计算机与信息技术学院 教师: 林友芳设断点断点标志断点标志主函数里只有两条语句,调用菜单函数后返回。北京交通大学计算
12、机与信息技术学院 教师: 林友芳设置断点对话框北京交通大学计算机与信息技术学院 教师: 林友芳去掉断点n把光标移动到给定断点所在的行,再次按F9就可以取消断点。n同前面所述,打开Breakpoints对话框后,也可以按照界面提示去掉断点。 北京交通大学计算机与信息技术学院 教师: 林友芳条件断点(有点难)n可以为断点设置一个条件,这样的断点称为条件断点。对于新加的断点,可以单击Conditions(条件)按钮,为断点设置一个表达式。n当这个表达式发生改变时,程序就被中断。北京交通大学计算机与信息技术学院 教师: 林友芳数据断点(有点难)n数据断点只能在Breakpoints对话框中设置。n选择
13、“Data”页,就显示了设置数据断点的对话框n。在编辑框中输入一个表达式,当这个 表达式的值发生变化时,数据断点就到达。n一般情况下,这个表达式应该由运算符和全局变量构成,n例如:在编辑框中输入 SelectTime这个全局变量的名字,那么当程序中有SelectTime+时,程序就将停在这个语句处。 北京交通大学计算机与信息技术学院 教师: 林友芳消息断点(有点难)nVC也支持对Windows消息进行截获。他有两种方式进行截获:窗口消息处理函数和特定消息中断。 5. 设了断点以后,如何走程序?北京交通大学计算机与信息技术学院 教师: 林友芳进入调试:1. 点此按钮2. 按F53. BuildS
14、tart DebugGo北京交通大学计算机与信息技术学院 教师: 林友芳黄色右箭头表示当前就要执行的代码行当前执行中的上下文(语境,函数)自动给出的当前语境下的变量值、返回值观察窗口Debug工具栏自定义的观察窗口,可以定义多组北京交通大学计算机与信息技术学院 教师: 林友芳掌握Debug工具条的主要按钮含义Stop debugging停止调试step into单步跟进,如果当前语句有函数调用,则单步进入函数执行,否则单步执行完一条语句。step over单步执行完当前语句,若当前语句有函数调用,除非被调用函数中有断点,否则不会跟进函数。step out执行完当前语句所在函数的执行,返回该函数
15、的调用处。Run to cursor执行到当前光标处,注意观察箭头便于理解记忆北京交通大学计算机与信息技术学院 教师: 林友芳常用进程控制快捷键快捷键快捷键说明说明F5goSHIFT+F5Step overCTRL+F5Execute programF7BuildCTRL+F7CompileF10Step overCTRL+F10Run to cursorF11Step intoSHIFT+F11Step outCRTL+SHIFT+F5Restart北京交通大学计算机与信息技术学院 教师: 林友芳点击Step into后,进入函数Menu()执行当前就要执行的代码行语境切换成Menu()6.
16、 如何了解执行状态?北京交通大学计算机与信息技术学院 教师: 林友芳查看数值nVC支持查看变量、表达式和内存的值。所有这些观察都必须是在断点中断的情况下进行。n观看变量的值最简单,当断点到达时,把光标移动到这个变量上,停留一会就可以看到变量的值。 北京交通大学计算机与信息技术学院 教师: 林友芳watchnVC提供一种被称为Watch的机制来观看变量和表达式的值。n在断点状态下,在变量上单击右键,选择Quick Watch, 就弹出一个对话框,显示这个变量的值。 北京交通大学计算机与信息技术学院 教师: 林友芳watchn单击Debug工具条上的Watch按钮,就出现一个Watch视图(Wat
17、ch1, Watch2 , Watch3,Watch4),在该视图中输入变量或者表达式,就可以观察 变量或者表达式的值。n注意:这个表达式不能有副作用,例如+运算符绝对禁止用于这个表达式中,因为这个运算符将修改变量的值,导致 软件的逻辑被破坏。 北京交通大学计算机与信息技术学院 教师: 林友芳点击Step over后,开始执行当前函数代码自动给出当前语境下变量的值当前执行的代码行北京交通大学计算机与信息技术学院 教师: 林友芳当前程序运行结果?什么也没有,为什么?因为没有输出任何内容北京交通大学计算机与信息技术学院 教师: 林友芳再次单击Step over,执行完第一个printf语句prin
18、tf语句返回值被自动给出,表示输出了38个字符执行结果?北京交通大学计算机与信息技术学院 教师: 林友芳执行第1条printf语句以后的输出结果北京交通大学计算机与信息技术学院 教师: 林友芳点击run to cursor将光标移动到此行printf语句返回值被自动给出,表示输出了38个字符北京交通大学计算机与信息技术学院 教师: 林友芳执行到此行之前刚才那几个变量怎么不见了?因为有点远(上下文),只显示执行位置附近的,想看怎么办?可以在此处增加你想看的东西。基本方法:直接在此处写上(还有别的办法加)你想看的内容的表达式如:想看SeleFun和SelectTime值,想看SelectTime的
19、地址,怎么办北京交通大学计算机与信息技术学院 教师: 林友芳手工增加SeleFun和SelectTime变量,系统会自动给出它们的值北京交通大学计算机与信息技术学院 教师: 林友芳继续以step over的方式到此语句,注意观察debug工具栏的几个按钮已经变灰,原因是在执行scanf语句,等待你在运行窗口中录入数据手工在任务条点击显示运行窗口,输入3北京交通大学计算机与信息技术学院 教师: 林友芳当前上下文的一些东西又自动出现了这个+号表示还可展开查看各个变量的值发生了变化北京交通大学计算机与信息技术学院 教师: 林友芳继续单击Step over,执行到ExecuteFunction函数,然
20、后单击Step into进入此函数。北京交通大学计算机与信息技术学院 教师: 林友芳单击Step into进入ExecuteFunction函数执行代码。为什么SeleFun的值发生了变化?原因:语境发生变化了,ExecuteFunction函数中没有这个变量。北京交通大学计算机与信息技术学院 教师: 林友芳点击此处继续执行当前执行的代码行等待用户输入手工在任务条点击显示运行窗口,输入任意正整数如:53北京交通大学计算机与信息技术学院 教师: 林友芳单击Step into进入prime函数执行代码。北京交通大学计算机与信息技术学院 教师: 林友芳点击step out,退出当前函数prime的执
21、行,返回到原调用处北京交通大学计算机与信息技术学院 教师: 林友芳Call stack调用堆栈n想要知道函数被调用的过程怎么办?n调用堆栈n反映了当前断点处函数是被那些函数按照什么顺序调用的。n单击Call stack按钮,显示Call Stack对话框。其中显示了一个调用系列,最上面的是当前函数,往下依次是调用函数的上级函数。n单击这些函数名可以跳到对应的函数中去。 此按钮北京交通大学计算机与信息技术学院 教师: 林友芳点击step out,退出当前函数prime的执行,返回到原调用处原调用处运行结果显示北京交通大学计算机与信息技术学院 教师: 林友芳再次点击step out,退出当前函数E
22、xecuteFunction的执行,返回到原调用处此时准备执行下一遍循环注意语境变化观察SeleFun值的改变,因为语境发生变化,返回到了主调函数Menu()。观察SelectTime值的变化,因为选择了一次执行北京交通大学计算机与信息技术学院 教师: 林友芳执行该行语句,等待用户输入选项北京交通大学计算机与信息技术学院 教师: 林友芳输入选项为0,准备结束程序运行北京交通大学计算机与信息技术学院 教师: 林友芳执行该语句,跳出for循环打印选择次数北京交通大学计算机与信息技术学院 教师: 林友芳Menu()执行完毕,执行return语句后结束整个程序的调试运行。在调试过程中可以随时点击此按钮
23、结束程序运行。8. Debug工具条上的其它按钮功能北京交通大学计算机与信息技术学院 教师: 林友芳Memory n 由于指针指向的数组,Watch只能显示第一个元素的值。为了显示数组的后续内容,或者要显示一片内存的内容,可以使用memory功能。n在 Debug工具条上点memory按钮,就弹出一个对话框,在其中输入地址,就可以显示该地址指向的内存的内容。 北京交通大学计算机与信息技术学院 教师: 林友芳Varibles nDebug工具条上的Varibles按钮弹出一个框,显示所有当前执行上下文中可见的变量的值。特别是当前指令涉及的变量,以红色显示。 北京交通大学计算机与信息技术学院 教师
24、: 林友芳Reigsters Debug工具条上的Reigsters按钮弹出一个框,显示当前的所有寄存器的值。 北京交通大学计算机与信息技术学院 教师: 林友芳其他调试手段 系统提供一系列特殊的函数或者宏来处理Debug版本相关的信息,如下: 宏名宏名/函数名函数名说明说明TRACE使用方法和printf完全一致,他在output框中输出调试信息ASSERT 它接收一个表达式,如果这个表达式为TRUE,则无动作,否则中断当前程序执行。对于系统中出现这个宏 导致的中断,应该认为你的函数调用未能满足系统的调用此函数的前提条件。例如,对于一个还没有创建的窗口调用SetWindowText等。VERIFY 和ASSERT功能类似,所不同的是,在Release版本中,ASSERT不计算输入的表达式的值,而VERIFY计算表达式的值。北京交通大学计算机与信息技术学院 教师: 林友芳本章应掌握的内容n掌握开发环境,工作空间,项目,程序模块的基本概念。n学会使用基本的调试工具。本部分结束