Delphi7调试工具的使用

Delphi7中提供了一些附加调试工具来帮助用户检查程序错误。从性能上讲,其中一些工具属于高级调试工具。尽管高级调试工具不像其他工具那样常用,但对于经验丰富的编程人员来说,它们是非常有用的。

Evaluate/Modify对话框(The Evaluate/Modify Dialog Box)

该对话框用于检查变量的当前值和修改变量的值。使用这个对话框,用户可以修改某个变量的值来测试不同的输出结果。在调试期间改变一个变量的值,使用户能测试程序不同参数的效果,而不必每次都重新编译。如下图主菜单【Run | Evaluate/Modify】或按【Ctrl + F7】

Note

该对话框的工具栏即可显示大图标也显示小图标,要在大小图标进行切换的话,用鼠标拖动位于工具栏下方的线条,工具就会在大小图标间进行调整了。

Evaluate/Modify对话框,类似于Watch List和Debug Inspector。如果点击代码中的一个变量,并从Code Editor的快捷菜单中选择【Evaluate/Modify】菜单项,该变量会自动被求值。也可手动输入要进行求值的变量,从主菜单选择【Run | Evaluate/Modify】菜单项,然后输入要求值的变量名。

Expression字段用于输入要对其求值的变量名或表达式。点击Evaluate按钮或按Enter键,表达式或变量就会被求值。结果显示在Result字段中。如下图:

Note

Evaluate/Modify对话框可当做一个简易计算器使用,,可输入十六进制数(或十进制数,或两者都可)的数学公式,让计算器计算结果。例如如下图:

还可在Expression中输入逻辑表达式,Result显示True。如下图:

程序必须停在断点处才能使用Evaluate/Modify对话框这一功能。

如果用户想要改变一个变量的值,可在New Value字段输入新值并点击Modify按钮,变量就会变成输入的新值。当用户点击Run按钮重启启动程序(或继续步进调试)时,就会使用新值。

Note

与Watch List和Debug Inspector不同的是,当用户一行一行执行代码时,Evaluate/Modify对话框不会自动更新。如果代码改变了Evaluate/Modify对话框中变量的值,必须再点击Evaluate按钮才能看到改变后的值。这使得Evaluate/Modify对话框有个优点:步进式调试代码时速度更快,因为调试器不必每执行一行代码就求一个表达式的值。可交互地使用这个对话框来求变量或表达式的值,并且求完值后立即关闭此对话框。

调用栈窗口(The Call Stack Window)

在程序运行阶段,用户可通过查看调用栈来检查应用程序调用的函数或过程。从主菜单上选【View | Debug Windows | Call Stack】菜单项来显示Call Stack窗口,该窗口列出应用程序调用的函数和过程,按调用顺序的先后排列,最近调用的函数或过程列在窗口顶部。

双击Call Stack窗口中的方法名,会显示该方法的代码行(如果该方法在应用程序中);当函数或过程的代码不存在时(VCL方法), Call Stack窗口显示该过程所在的模块名和地址。双击没有源代码的函数和过程,会显示CPU窗口。

当碰到Windows Access Violation错误时,查看调用栈非常有帮助。通过查看调用栈,用户能找出程序中产生错误的位置;找到程序中错误位置是排除错误的第一步。

Tip

如果调用栈列表中包含有无意义的信息,那可能是调用栈被破坏了。被破坏的调用栈通常标志着栈溢出或内存覆盖。

CPU窗口(The CPU Window)

可从主菜单【View | Debug Windows | CPU】菜单项或按【Ctrl + Alt + C】来显示CPU窗口。

CPU窗口使用户能够在汇编指令级查看应用程序。可以执行一个汇编指令就暂停一下;也可以运行程序至某条汇编指令(就像用常规调试器运行程序至某行源代码一样)。CPU窗口分成五个窗格:反汇编窗格(Disassembly pane)、寄存器窗格(CPU Registers pane)、标志窗格(Flags pane)、原始栈窗格(Machine Stack pane)和转储窗格(Memory Dump pane)。

每个小窗格都有一个快捷菜单,这个快捷菜单提供使用该窗格时所需的功能。要有效地使用CPU窗口,必须懂汇编语言。显然,CPU窗口是一个高级调试工具。

Go to Address命令(The Go to Address Command)

Go to Address命令也是一个高级调试工具。当应用程序出现故障时,Windows会显示一条出错消息,给出违例地址。用户可以使用Go to Address命令来找出程序中发生故障的位置。当Windows发出Access Violation出错消息时,会显示一个类似于下图的错误。

当用户看到这样的出错消息时,记下违例发生的地址,从Code Editor快捷菜单中选择【Debug | Go to Address】菜单项来显示Go to Address对话框。

把记下的地址输入到Go to Address对话框的Address字段中。点击OK,调试器就开始查找发生故障的源代码行。如果故障出现在代码中,则光标会指着产生错误的源代码行。如果是代码以外的地方发生故障,会弹出一个消息框告诉用户,代码中未找到该地址。这是一个高级调试工具,用户可能还从未使用过。

**************************************************

Delphi在调试的时候查看变量的值

Delphi调试概述

除非你的程序只有几行,否则几乎不可能一次写成,因此调试就非常必要。然而许多初学者不知道如何进行调试,写完程序就运行,发现结果不对再看代码,这样觉得非常吃力。这里,简单介绍一下Delphi所提供的调试功能。

1. 语法检查(Syntax Check)

Delphi提供了语法检查的功能,这个功能和正常编译很相似,同样也会给出各类错误和警告信息,但是不会生成代码。

Delphi的编译信息分为4类:Fault(故障),Error(错误),Warning(警告)和Hint(提示)。Fault是指导致编译程序不能继续工作的错误,例如内存溢出等;Error是指发现用户程序不符合某些规定而导致不能按照用户程序的要求进行处理;Warning是指用户程序使用了某些不符合规定的形式,但是并不影响编译程序生成目标文件;Hint是指编译程序对用户程序的某些形式提出了怀疑。

前两类信息是必须要解决的,否则你不能运行你的程序,但是往往会有很多人忽略后两种信息。然而,这些信息却是非常重要的。

2. 启动、暂停、中止集成调试程序

最基本、最重要的调试手段包括:单步跟踪、断点、变量观察、堆栈检查等。所有这些功能在Delphi的集成调试程序中都能提供。

当你按下F9(Compile and Run,编译并运行)一个程序时,就已经启动了Delphi的集成调试程序,而按下Ctrl+Break(Program Pause,程序暂停)时则会暂停被调试程序返回到集成调试程序中去,再次按下F9会从暂停地地方继续执行,而Alt+F2(Program Reset,程序复位)则会完全中止被调试程序的执行,返回集成调试程序中去。

3 单步跟踪(Step)

所谓单步跟踪是指一行一行地执行程序,每执行一行语句后就停下来等待指示,这样你就能够仔细了解程序的执行顺序,以及当时的各种状况。

注意:虽然Object Pascal允许在一行内书写任意多的语句,但是所有的单步跟踪都以“行”为单位,因此为了便于调试,主张在一行内只写一条语句,否则会给你带来很大的麻烦。

单步跟踪可以分为Step Over(跳过)、Trace Into(跟踪进入)和Trace to Next Source Line(跟踪到下一条源代码行)。

Step Over和Trace Into都是执行一行语句,差别在于遇到过程和函数时Trace Into将会进入过程和函数,而Step Over不会,而只会把过程和函数作为一条语句执行。

当使用Ctrl+Break暂停程序时,程序不一定停在你的源代码位置上,而可能是在操作系统或者其它模块中,此时集成调试程序会出现一个CPU窗口(CPU Window),用汇编指令的形式显示当前的内容,可以用Trace to Next Source Line继续执行程序,直到程序执行到第一条有源代码的地方。

4 断点(Breakpoint)

断点是调试中非常重要的一个手段。由于在执行到某些代码前需要执行许多其它代码,不可能用单步跟踪一条一条执行过来,这时只要在需要暂停的地方设置一个断点,然后让程序运行,当执行到这个断点位置时不需要用户干预就会暂停并返回集成调试程序。

Delphi提供了丰富的断点功能,包括:源代码断点、指令断点、数据断点等。

源代码断点(Source Breakpoint)是指在你的源程序中设置断点,指令断点(Address Breakpoint)是指在某机器指令处设置断点,数据断点(Data Breakpoint)是指当写入某变量时暂停用户程序。

所有的断点都可以设置更详细的属性,包括:条件、通过次数、组、高级操作等。

条件(Condition)是指触发断点的条件,例如你可以写:a=10,表示当a等于10时在这个断点位置暂停;

通过次数(Pass Count)是指即使符合条件,也需要执行这些次数才N暂停,例如在某断点设置通过次数为5,则表示当第5次通过这个断点时才暂停程序,当然,如果有条件存在的话还要符合相当次数的条件;

组(Group)是指一组断点,你可以用一个名字来标记许多断点,这样你可以用禁止或允许组(Disable Group/Enable Group)来同时打开或禁止多个断点。

高级操作是指和每个断点相关的一些行为(Action),具体如下:

中断(Break):中断程序,这是默认操作。

忽略后续异常(Ignore subsequent exceptions):通过这个断点后忽略所有异常(exceptions);

处理后续异常(Handle subsequent exceptions):通过这个断点后处理异常,这和前一个操作是对应的;

记录信息(Log message),通过这个断点时记录一条事件日志信息,你可以在事件日志(Event Log)中查看这条信息;

表达式求值(Eval expression):对指定的一个表达式进行求值,并且可以通过记录结果(Log result)把这个结果记录在日志中;

禁止/允许组(Enable group/Disable group):通过这个断点以后禁止或者允许其它的组,由此可以控制其它断点的状态。

在Delphi中除了上述的显式断点以外,还提供了隐式断点:运行到光标(Run to cursor)和运行到返回(Run until return)。

运行到光标是让程序到当前光标所在程序行,相当于你在当前光标位置设置了一个断点。这是一次性断点,并且如果在到达这里前遇到了其它断点,会停止在那个断点的地方,同时取消了这个临时断点。

运行到返回是用于过程和函数中,运行到过程和函数退出的位置,使得可以迅速返回上层调用程序。

5. 变量查看(Watch)/检查(Inspect)

在程序暂停的时候你可以用Watch查看某个变量,按Ctrl+F7(Add Watch,添加查看)可以在查看列表(Watch List)中增加一个变量。在Watch中你可以查看变量或者表达式,指定数据的格式,甚至可以指示Delphi调用某些函数,显示函数的返回值。

有一种快速查看模式,称为Local Variables(局部变量),按Ctrl+Alt+L能够显示这个窗体,里面是当前过程或函数的局部变量。

Delphi还支持一种临时的求值模式(Evaluate/Modify),按Ctrl+F4显示求值框,你可以在这里输入一个变量或者表达式,计算其数值,对于变量还可以在运行时改变它的值,这样如果你已经发现数据有错,你可以修改它,让程序继续运行下去,就像这个数值就是程序得出的一样。

检查(Inspect),是一种可以进一步查看变量信息的手段。把光标放在某个变量前,按Alt+F5显示检查窗。在这里可以看到有关这个变量的详细信息,包括:类型、值等,这对于类类型、记录类型尤其有用。和Evaluate/Modify一样,你也可以改变这些值。

6. 调用堆栈(Call Stack)

对于某些递归调用和复杂的嵌套调用来说,使用Call Stack功能能够方便的检查函数的调用情况。

按Ctrl+Alt+S可以显示这个窗体,在最上面的是当前过程或函数,在最下面的往往是你的主程序。例如:

TForm1.Button1Click(???);

Project1

这表示Project1调用了方法TForm1.Button1Click,由于其参数是一个对象(Sender:TObject),不能求值,所以用???表示。双击Project1可以看出在什么地方调用了TForm1.Button1Click(如果调用点没有源代码,则显示有源代码的第一行)。

7. 高级调试功能

上面所说的是常规的调试功能,Delphi还提供了很多高级调试功能。

线程状态(Thread Status):显示当前程序中有多少线程在运行,各线程的状态是什么?参数是什么?

模块(Modules):显示当前进程使用了多少模块,其名称和地址是多少?这对于调试DLL时很有用。

CPU/FPU:在汇编语言层次显示代码,这能够更加精确地观察程序是如何运行的,各寄存器是怎么变化的。

进程附着(Attach Process):为了调试某些特殊程序(例如Windows 2000下的服务【Service】),允许先运行用户程序,再运行调试程序。

远程调试(Remote Debug):允许在一台计算机上运行用户程序,在另外一台计算机上运行Delphi,通过网络进行调试,这对于调试大型程序很有用,也能调试那些对系统有特殊要求的程序。

Delphi变量查看(Watch)/检查(Inspect)

在程序暂停的时候你可以用Watch查看某个变量,按 Ctrl+F7(Add Watch,添加查看)可以在查看列表(Watch List)中增加一个变量。在Watch中你可以查看变量或者表达式,指定数据的格式,甚至可以指示Delphi调用某些函数,显示函数的返回值。

有一种快速查看模式,称为Local Variables(局部变量),按 Ctrl+Alt+L 能够显示这个窗体,里面是当前过程或函数的局部变量。

Delphi还支持一种临时的求值模式(Evaluate/Modify),按 Ctrl+F4 显示求值框,你可以在这里输入一个变量或者表达式,计 算其数值,对于变量还可以在运行时改变它的值,这样如果你已经发现数据有错,你可以修改它,让程序继续运行下去,就像这个数值就是程序得出的一样。

检查(Inspect),是一种可以进一步查看变量信息的手段。把光标放在某个变量前,按Alt+F5显示检查窗。在这里可以看到有关这个变量的详细信息,包括:类型、值等,这对于类类型、记录类型尤其有用。和Evaluate/Modify一样,你也可以改变这些值。