除非你的程序只有几行,否则几乎不可能一次写成,因此调试就非常必要。然而许多初学者不知道如何进行调试,写完程序就运行,发现结果不对再看代码,这样觉得非常吃力。这里,简单介绍一下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一样,你也可以改变这些值。