




已阅读5页,还剩14页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
DELPHI的编译指令$IFDEF WIN32 - 这可不是批注喔!对于Delphi来说左右大括号之间的内容是批注然而$(左括号后紧接着货币符号)对于Compiler(编译器)而言并不是批注而是写给Compiler看的特别指示。应用时机与场合Delphi中有许许多多的Compiler Directives(编译器指令)这些编译指令对于我们的程序发展有何影响呢? 它们又能帮我们什么忙呢?Compiler Directive 对程序开发的影响与助益, 可以从以下几个方向来讨论:协助除错版本分类程序的重用与管理设定统一的执行环境协助除错稳健熟练的程序设计师经常会在开发应用系统的过程中特别加入一些除错程序或者回馈验算的程序这些除错程序对于软件品质的提升有极其正面的功能。然而开发完成的正式版本中如果不需要这些额外的程序的话要想在一堆程序中找出哪些是除错用的程序并加以删除或设定为批注不仅累人而且容易出错况且日后维护时这些除错程序还用得着。此时如果能够应用像是$IFDEF的Compiler Directives 就可以轻易的指示Delphi要/不要将某一段程序编进执行文件中。同时Compiler本身也提供了一些错误检查的开关可以预先对程序中可能的问题提醒程序设计师注意同样有助于撰写正确的程序。版本分类除了上述的除错版本/正式版本的分类之外对于像是试用版普及版专业版的版本分类也可以经由Compiler Directive的使用为最后的产品设定不同的使用权限。其它诸如中文版日文版国际标准版等全球版本管理方面同样也可以视需要指示Delphi特别连结哪些资源档或者是采用哪些适当的程序。以上的两则例子中各版本间只需共享同一份程序代码即可。Delphi 1.0 与 Delphi 2.0有许多不同之处组件资源文件(.DCR)即是其中一例两者的档案格式并不兼容在您读过本文之后相信可以写出这样的程序指示Delphi在不同的版本采用适当的资源文件以利于组件的安装。$IFDEF WIN32$R XXX32.DCR$ELSE$R XXXX16.DCR$EDNIF程序的重用与管理经过前文的讨论后相信你已经不难看出Compiler Directives在程序管理上的应用价值。对于原始程序的重用与管理也是Compiler Directives 使得上力的地方. 举例来说:Pascal-Style字符串是Delphi 1.0与 Delphi 2.0之间的明显差异除了原先的短字符串之外Delphi 2.0之后还多了更为方便使用的长字符串同时系统也额外提供了像是 Trim() 这样的字符串处理函式。假如您有一个字符串处理单元必须要同时应用于Delphi 1.0 与 2.0的项目时编译指示器可以帮你的忙。此外透过像是$I xxxx 这样的 Compiler Directives我们也可以适当的含入某些程序, 同样有助于切割组合我们的程序或编译设定。设定一致的执行环境项目小组的成员间必须有共同的环境设定我很难预料一个小组成员间彼此有不同的$B$H$X设定最后子系统在并入主程序时会发生什么事。此外, 当您写好一个组件或单元需要交予第三者使用时, 使用编译指示器也可以保证组件使用者与您有相同的编译环境。使用Compiler Directives指令语法Compiler Directives从外表看起来与批注颇为类似, 与批注不同的是:Compiler Directives的语法格式都是以$开始, 不空格紧接一个名称(或一个字母)表明给Compiler的特别指示, 再加上其它的开关或参数内容, 最后以右大括号作为指令的结束, 例如:$B+$R-$R MyCursor.res同时, 就如同Pascal的变量名称与保留字一样, Compiler Directives也是不区分大小写的。从指令的语法格式来说Compiler Directives可以进一步分类成以下三种格式:开关指令(Switch directives)这类指令都是单一字母以不空格的方式连接+或-符号; 或者是开关名称以一个空格后连接ON或OFF来表示作用/关闭某一个编译指示开关。例如:$A+$ALIGN ON开关型的编译指令不一定要分行写, 它们可以组合在同一个编译指示的批注符号之间, 但必须以逗号连接, 而且中间不可以有空格, 例如:$B+,H+,T-,J+光标停留在程序编辑器的任一位置时按下Ctrl+O O, 完整的Compiler Directives将会全部列于Unit的最上方。参数指令(Parameter directives)有些Compiler Directives需要在编译名称后面连接自定的参数(文件名称或指定的内存大小), 例如: $R MyCursor.res, 即在指示Delphi在编译连结时, 含入MyCursor.res这个资源档。条件指令(Conditional directives)指示Compiler在编译的过程中, 按我们设定的条件, 选择性的采用/排除不同区域的程序代码。以下是一个条件编译的例子, 第一与第三列是写给Compiler看的,指示 Compiler在 _DEBUG这个条件名称完成定义的情况才编译ShowMessage()这列程序;反之, 如果 _DEBUG 当时没有定义的话, 这段程序几乎与批注无异, Compiler对它将视而不见。$IFDEF _DEBUGShowMessage(IntToStr(i);$ENDIF如何从IDE改变Compiler directives设定从Delphi的IDE程序整合发展环境, 我们很方便的就可以修改各个compiler directives的设定, 方法是: 从Delphi IDE主选单: Project/Options/Compiler, 直接核选/取消各个CheckBox。值得注意的是, 改变一个项目的Compiler directives并不会影响其它的项目, 换言之, 各个项目都保有自己一套编译指示。假如您希望其它的项目也采用相同一套的Compiler directives, 在上述ProjectOptions对话盒的左下方有一个Default选项, 选取这个CheckBox之后, 虽然对于既有的项目没有作用, 但未来新的项目都将可以采用这组设定作为默认值。将Compiler directives写入程序透过Delphi的整合环境设定Compiler directives的确十分简便, 但是许多情况下我们仍然需要将Compiler directive直接加到程序中。至少有两个原因支持我们这么作:局部控制编译条件在Project/Options/Compiler中所作的设定, 影响所及是整个项目, 如果某一段程序要特别使用不同的编译设定, 就必须直接将编译指示加到程序中。下列这段取自Online Help的程序范例, 即应用了$I编译指令局部控制在发生I/O错误时不要举发例外讯息, 这样, 我们就可以编译出一支在这段程序区域中不会产生I/O例外讯息的档案侦测函数。function FileExists(FileName: string): Boolean;varF: file;begin$I-AssignFile(F, FileName);FileMode := 0; ( Set file access to read only Reset(F);CloseFile(F);$I+FileExists := (IOResult = 0) and (FileName );end; FileExists 程序的可移植性我们都可能会用到其它公司或个人创作的unit或component, 也可能分享程序给其它人, 换句话说, 单元或程序可能会在不同的机器上编译, 直接将Compiler directives加入程序, 不仅可以免去程序使用前需要特别更改IDE的麻烦, 更重要的是解决了各个单元间要求不同编译环境的歧异。注意事项Compiler directives的作用与影响范围如同变量的可见范围与生命周期, 在我们使用 Compiler Directives 时也必须注意各个Compiler Directives 的作用范围.Compiler Directives的作用范围可分为以下两种:全域的全域的Compiler Directives, 影响所及是整个项目; 我们稍早前提到经由Delphi IDE改变Compiler directives的方式就属于全域的设定。区域的而区域的Compiler Directives 影响所及只从Compiler Directives 改变的那一行开始, 直到该程序单元(Unit)的结束或另一个相同的Compiler Directives 为止, 对其它的程序单元并没有影响。也就是说, 如果在unit中特别加入Compiler directives, Compiler会优先采用区域的设定, 然后才是属于项目层级的全域设定。值得一提的是, 在程序中直接加入Compiler directives的最大作用范围也只限于当时那个单元而已, 对其它单元并没有任何影响, 即使是以uses参考也是一样。也就是说, 我们可以透过uses参考其它unit公开的变量与函式, 但是各个unit的编译指令并不会互相参考。这项独立的性质, 使得unit之间编译环境的设定与关系变得十分简洁, 例如Delphi 2.0的VCL都是在$H+的情况下编译的, 因此, VCL中的字符串都是以长字符串的型态编译而成的, 有了这项编译指令独立的特性, 不论我们Prject中的设定为何, 这些在VCL中定义过的字符串都是长字符串。我们的Project也不会因为uses了VCL中的unit而改变了自己的设定。因此, 在我们移交程序到网络上时, 大可以放心的在程序中加入必要的Compiler directives, 别担心, 即使别的unit以uses参考了我们的程序, 也不影响它自己原来的设定。如果我们自行以$DEFINE _DEBUGVERSION($DEFINE在稍后的个别指令介绍中将有说明)定义了一个条件符号, 这个新的条件符号也是区域的, 换句话说, 它只从定义的那一个单元的那一列之后才成立, 当然, 也只对目前这个单元有效.由于自订的条件符号只有区域的作用, 如果有好几个程序单元都需要参考到某一个条件符号, 怎么办呢? 嗯! 在-一个程序单元开头处中都加上编译指示是最直接的方式, 可是略嫌麻烦, 特别是编译指示有变时, 要一一修正各个单元的设定内容, 很容易因为疏忽而出错。比较简易可行的作法是从Delphi IDE整合发展环境的主选单-Project / Options/ Directories/Conditional的 Conditionals 中填入条件名称。这样, 相对于项目的各个unit而言, 就有了一个全域的条件符号。或者, 您也可以参考本文对于$I这个Compiler Directive的说明。 我在那里指出了另一个弹性的解决方式。修改过编译指令后, 建议Build All过一次程序.请试一试这个程序:procedure TForm1.Button1Click(Sender: TObject);begin/ ifopt是用来侦测某一个编译开关的作用状态$ifopt H+ShowMessage(H+);$elseShowMessage(H-);$endifend;在我们执行上述程序时, 在Delphi预设的是$H+时, ShowMessage()会在画面上会显示H+, 执行过后, 让程序与form的内容与位置保留不变, 单纯的从主选单: Project/Options/Compiler, 将Huge Strings的核对方块清除($H-), 然后按下F9执行,咦! 怎么还是看到H+?!那是因为Delphi只会在unit内容经过异动后才会重新将.PAS编译成.DCU, 在我们的例子中, 程序并没有变动, .DCU当然也没有重新产生, 最后.EXE的这个部分自然也是没什么变化。所以, 要解决这个问题, 只要以Delphi IDE主选单Project/Build All指示Delphi重新编译全部的程序即可。因此, 如果您从Delphi IDE修改过Compiler Directives后, 记得要Build All喔!不应该用来作为程序执行流程控制在程序中, 我们可以使用if叙述, 根据执行当时的情况控制程序执行时的流程, 但我们不可以用$IFDEF来作同样的事, 为什么? 从上述的说明, 相信您不难发现, Compiler directives会对最后.EXE的内容发生直接的影响, 应用像是$IFDEF指示Compiler的结果, 几乎可以视同授权Compiler在编译的那个时候自动选用/舍弃程序到.DCU, .EXE中, 换句话说, 在程序编译完成时, 会执行到那一段程序已成定局了, 我们自然不能用它来作程序流程的控制。条件编译的巢套最多可以16层在使用$IFDEF$ENDIF条件编译我们的程序时, 一个$IFDEF中可以再包含另一个$IFDEF, 但深度最多只能16层, 虽然是个限制, 但以正常的情形来说, 这应该已经足够了。有些Compiler directives不应写在Unit中对于像是$MINSTACKSIZE$MAXSTACKSIZE管理堆栈大小, 或者像是$APPTYE指示程序编译成图形/文字模式的Compiler directives, 只能写在.DPR中, 写在Unit中是没有效果的。建议事项确定您了解指令的影响由于编译指令的影响是如此直接与深远, 在修改与应用某一个Compiler directive时, 请确定您已经了解其含意与影响。打开全部的侦错开关Delphi有关侦错的Compiler directives如下:$HINTS ON$D+$L+$Q+$R+$WARNINGS ON各指令的用法您可以参阅本章稍后对个别指令的说明, 全部打开这些开关吧! 这样不仅让您可以使用Delphi IDE的除错器, 对于程序编译与执行过程中的问题, Delphi也会适时的反应, 有助于写作正确的程序。此处有一个迷思有待澄清-加入Dubug信息会不会让执行文件变大变慢啊?, 不一定。对于们像是$D+, $L+, $HINTS ON这些开关, 打开后, Delphi在编译时的确会额外加入一些除错信息, 使得.DCU的档案变大, 对于.EXE的档案大小并没有影响; 同时, 程序的执行速度也没有改变, 还可以应用IDE的除错器trace我们的程序, 值得应用。对于像是$Q, $R等Compiler directive, 的确会影响执行文件的大小与速度, 然而这并不动摇我们在研发期间使用它们的决定, 请想想看, 值得为这一点点的速度放弃程序的正确性吗? 当然, 程序开发完成后, 正式出货的版本, 可以关闭这两个开关。如果您写好了一个组件, 而且只预备提供.DCU, 由于没有.PAS可供Delphi IDE的Debugger追踪程序, 除错开关反而应该在组件脱手前关闭并重新编译.DCU, 否则会引起使用者那边找不到档案的例外讯息。善用$I$I FileName是一个非常有用的Compiler directive.应用这个指令, 我们可以弹性的管理Compiler directive的设定。条件名称请加入前导符不知道您有没有这个疑问 - 如果用$DEFINE定义的条件名称与变量名称相同时会发生什么事?procedure TForm1.Button1Click(Sender: TObject);varTEST: integer;begin$DEFINE TEST$IFDEF TESTShowMessage(Test);$ENDIFend;以上的程序编译与执行都没有问题, 但条件名称与变量名称重复毕意容易让人混淆, 因此, 假如能适当的为编译条件名称之前加上诸如底线(_TEST), 程序会比较容易阅读。设定一致的编译环境在您了解了Compiler Directives之后, 请立即开始着手修改您IDE中有关编译指示的各个开关并且设为Default, 这样, 日后您的项目乃至整个研发小组都将拥有共同一致的编译环境, 对于写出来的程序会以何种方式编译连结都了然于胸, 直接有助于子系统顺利并入主系统中。个别指令说明有了之前对于Compiler directives的观念之后, 接下来的这一节我将一一介绍几个常用的Compiler Directive的用法与注意事项, 您可以从这一节中学到更多有关Compiler directives的知识与使用细节。$A+ 字段对齐在$A+(默认值)的情形下, 如果没有使用 packed 修饰词宣告的 record 型态,其字段会以CPU可以有效存取的方式向 1. 2. 4 等边界对齐, 以获取最佳的存取速度。以下列的程序示例来说:$A+typeMyRecord = recordByteField: byte;IntegerField: integer;end;procedure TForm1.Button1Click(Sender: TObject);beginShowMessage(IntToStr(SizeOf(MyRecord);end;ShowMessage在$A+时显示的结果是:8; 倘若是$A-, 那所得的结果是5, 按理说, Byte应该只要一个byte就足够了, 但是考虑到硬件的执行特性, 经过对齐后的record会有比较好的执行速度。有关这个Compiler Directive要注意的事项是: 不管$A的开关是ON或OFF, 使用packed修饰过的记录宣告, 是一定不会对齐的. 例如:MyRecord = packed record / 不会对齐的记录宣告方式$APPTYPE GDI 应用程序型态一般的情形下, Delphi会以$APPTYPE GUI的方式产生一个图形的使用者接口程序, 如果您需要产生一个文字屏幕模式的程序, 那可以经由:在.DPR中加入$APPTYPE CONSOLE从主选单: Project/Options/Linker/EXE and DLL Options, 核取Generate Console ApplicationCheck Box。其它有关这个Compiler Directive的注意事项有:$APPTYPE不能应用在DLL的项目或单一的程序单元(Unit), 它只对.EXE有意义。而且只有写在.DPR中才有作用。我们可以应用System程序单元中的IsConsole函数在程序执行时侦测应用程序的类型。参阅Object Pascal手册第十三章可以知道更多有关Console Mode Application的信息。$B- 布尔评估请看以下的程序:if (Length(sCheckedDateString) 8)or EmptyStr(sCheckedDateString)or (sCheckedDateString = . . )or (sCheckedDateString = / / ) thenbeginResult := True;Exit;end;假如sCheckedDateString的字符串内容是85/12/241(长度9)的话, 以上的if述句, 其实在第一个逻辑判断时就已经知道结果了, 即使不看后来的逻辑运算结果也知道整个式子会是真值。假如您希望对整个逻辑表达式进行完整的评估 - 尽管结果已知, 后来的逻辑运算也不影响整个的结果时仍要全部评估过, 请将这个Compiler directives设为$B+, 反之, 请设为$B-, 系统的默认值是$B-。$D+ 除错信息当程序以$D+(默认值)编译时, 我们可以用Delphi整合发展境境的Debugger设定 断点, 也可以使用Trace Into或Trace Info追踪程序的执行过程, 值得注意的 是, 以$D+编译的程序, 执行的速度并不会受到影响, 只不过编译过的DCU的档 案长度会加大, 但EXE档的大小不变。$DEFINE条件名称 定义条件名称随着您对Compiler Directives的了解与应用程度的加深, 您会发现这是一个非常 实用的编译指示。经常, 我们会因为除错需要区别不同版本等缘故, 希望选择性的采用或排除某一段程序, 这个时候, 我们就可以先以$DEFINE定义好一个条件名称(Conditional name), 然后配合$IFDEF条件名称$ELSE$ENDIF指示编译器按指定的 条件名称之有无来选择需要编译的程序。以下列的程序片断来说:$DEFINE _ProVersionprocedure TForm1.Button1Click(Sender: TObject);begin$IFDEF _ProversionfrmPrint.ShowModal; / A$ELSEShowMessage(很抱歉, 试用版不提供打印功能);$ENDIFend;编译器将会选择编译上述A的那列程序, 日后, 如果我们需要编译简易版的程序版本时, 只要:将$DEFINE _ProVersion那列整个删掉。或者, 将$DEFINE _ProVersion改成-$DEFINE _ProVersion, 让它变成普通 的批注或者, 在$DEFINE _ProVersion的下一列加上$UNDEF _ProVersion, 解除_ProVersion这个条件名称的定义。这样, 由于_ProVersion这个条件名称未定义的缘故, Compiler就只会选择$ELSE下的那段程序, 重新编译一次, 不需费太多力气, 很容易的就可以制作出简 易版了, 省去了要同时维护两份程序的麻烦。使用$DEFINE时的其它注意事项如下:以$DEFINE定义的条件名称都是区域的。换句话说, 它的作用范围只在当时所 在的单元才有效, 即使定义在unit的interface, 由其它的unit以uses参考也没有 效, 仍然只有在目前的unit有作用。此外, 它的作用范围是从定义起, 到unit结尾或者以$UNDEF解除为止。如果程序单元中已经用$DEFINE定义了一个条件名称, 而且也没有用$UNDEF解除定义, 重新$DEFINE一个同样名称并没有作用, 换句话说, 它们是同一个.假如需要一个全域的条件名称, 您可以:主选单: Project / Options / Directories/Conditional 的 Conditionals 中填入条件名称。以下的标准条件名称, 是Delphi 2.0已经预先预备好的, 我们可以直接引用, 同时, 它们都是全域的, 任何Unit都可以参照得到。VER90: Delphi Object Pascal的版本编号。90表示9.0版, 日后若出现9.5版时 , 也会有VER95的定义。WIN32: 指出目前是在Win32(95, NT)作业环境CUP386: 采用386(含)以上的CPU时, 系统会提供本条件名称。CONSOLE: 此符号会于应用程序是在屏幕模式下编译时才定义。$DESCRIPTION 描述内容应用$DESCRIPTION可以指定加入一段文字到.EXE或.DLL表头的模块描述进入点 (module description entry)中通常我们会用这个Compiler Directive加入应 用程序的名称与版本编号到.EXE中。例如:$DESCRIPTION Dchat Version 1.0$X+ 扩充语法这是为了与之前的Pascal版本前向兼容的编译指令, 虽然设定这个开关型的指令仍有作用, 但笔者建议您大可保留系统的默认值$X+, 在$X+下:不需要非得准备一个变量接受函数的传回值, 换句话说, 函数的传回值可以舍 弃, 此时, 就可以像是呼叫程序一样, 很方便的呼叫函数。支持Pchar型态与零基的字符数组作为C语言以Null结尾的字符串。$HINTS OFF 提示讯息打关$HINTS开关后, Compiler会提示程序设计师注意以下的情况:变量定义了却没有使用程序流程中不会执行的for或while循环只有存入没有取用的指定叙述。意思是说, 指定数据到某一个变量之后, 却没 有任何的程序参考取用这个变量值。$HINTS ONprocedure MyTest;const _False = False;varI, J: integer;beginif _False thenfor I := 1 to 3 do ;J := 3;end;$HINTS OFF由于程序简单, 在两个$HINTS中间的程序, 我们不难看出:for循环不会执行到, I变量也因此不曾用过J := 3写了等于白写但在程序越写越长而日趋复雓时, 藉由$HINTS ON的协助, 比较容易察觉出程序 的毛病。$IFDEF $IFNDEF请参阅$DEFINE的说明, 在此补充说明$IFNDEF, 以下列程序来说, 即在指示 Compiler在_Test未定义时, 条件编译ShowMessage()那列程序:$IFNDEF _TESTShowMessage(_TEST not define);$ENDIF换言之, $IFNDEF相当于$IFDEF的$ELSE部分。$IFOPT 开关到底$B是开着或关着呢? 如果我们想要指示Compiler按照某一个编译开关当时的状态作我们指定的事, 应该该怎么做呢? 这时, $IFOPT就派得上用场了。例如:$R+$Q- / 特别指定为Q-$IFOPT R+ / 如果 Range Check 是开启的话ShowMessage(程序是在 Range Check 开启状态下编译的); / 这个 Q+ 也会在 IFOPT R+ 成立时才通知 Compiler$Q+$ENDIF$IFOPT Q+ShowMessage(Q 也变成开启状态了);$ENDIFShowMessage() 与 $Q+会在$R+ 的情形下才编译, 因此, 虽然我们事前特别指 示为$Q-, 第二个的ShowMessage()在程序执行时也可以看到Q 也变成开启状 态了。$IMAGEBASE 档案基础地址这个Compiler directive用来指示.EXE或.DLL加载时的预设地址。例如: $IMAGEBASE $00400000。如果指定加载的地址空间之前已经有其它模块占用了, Windows会为.EXE重新配置一个新的加载地址。对于.DLL来说, 如果可以成功配置到我们写在$IMAGEBASE的地址, 由于不需要重新配置内存地址, 不仅加载的速度 较快, 如果有其它程序也参照到这个DLL的话, 也可以减少加载时间与内存的消 耗。使用这个Compiler directive时需要注意的事项有:指定的叙述必须是一个大于$00010000的32位整数数值, 同时, 较低位置的16个位必须是零。DLL的建议地址范围从$40000000到$7FFFFFFF, 该范围的地址可以同时适用于Windows 95与Windows NT。$I文件名称 含入档案以Delphi IDE修改Compiler directives的确相当方便, 但往往我们仍然需要将Compiler directives直接加入程序中, 可是当我们这样作之后不用多久, 就会发 现要一一重新修改各个单元中的这些Compiler directives时, 实在是既无聊而又 容易出错的工作。这时候, 假如您一开始就采用$I文件名称, 整件事就会变得很简单。怎么做呢? 让我用一个例子告诉您 -先用一般的文书编辑器建好一个MySet.inc的普通文本文件, 内容为:$H+$DEFINE _Proversion在我们的程序中, 加入一列$I MySet.inc, 例如:unit Unit1;$I MySet.incinterfaceimplementation$R *.DFMprocedure TForm1.Button1Click(Sender: TObject);begin$IFDEF _ProVersionShowMessage(专业版);$ElseShowMessage(只有专业版才有此功能);$ENDIFend;这是子程序的观念嘛! 没错, 就是这么简单而已, 以后如果有任何变化, 修改MySet.INC, 然后Project/Buile All即可, 实在是够简单的了。基本动作会了之后, 让我告诉你多一点有关$I文件名称的事。一旦应用了$I文件名称, 几乎等于Compiler在编译时, 让Compiler将这个档 案的内容贴进我们的程序中的那个位置。如果没有注明扩展名, Delphi预设这个档案是.PAS。如果在项目的目录中找不到这个档案的话, Delphi会陆续搜寻Tools/Options/Library中的Library Path中的目录。另外, 当您写作了一个DLL, 使用者在使用其中的函数前必须宣告过, 如果能够一并提供这些函数的宣告文件, 使用者只要一行$I xxx即可, 是不是很方便呢?$I+ EInOutError检查在$I+(系统默认值)状态编译的程序, 一旦发生I/O错误时, 将会举发一个EInOutError的例外, 假如我们在特定的情况下不希望出现这个例外的讯息时(例如前文提到的侦测档案是否存在函数), 可以将这个Compiler directive设为$I-, 此时, 程序执行时是否发生过错误,程序设定师必须自行检查IOResult这个公用变 数的值, 如果是零, 表示没有错误, 非零的错误代码含意请详查Online help。$L文件名称 连结目标文件如果您有一个.OBJ文件要并入Delphi的程序时, 可以在程序中加入:$L OTHER.OBJ这样, 就可以使用OTHER.OBJ中的程序了, 值得注意的是, 函数或程序在呼叫前,仍然必须用external宣告过, 表明这些模块是来自外部的函式。举例来说, 笔者有一份由Keypro厂商提供的.OBJ档, 在使用时, 相关的程序如下:$L hasptpw.obj$F+procedure hasp (Service, SeedCode, LptNum, Pass1, Pass2 : word;var p1,p2,p3,p4 : word); external;$F-经过$L hasptpw.obj宣告之后, 程序的其它部分就可以直接呼叫原先位于 hasptpw.obj中的hsap这个程序了。$L+ 区域符号信息在$L+时, Delphi会额外加入一些区域符号信息, 这使得我们可以应用Delphi IDE中的View/Call Stack, View/Watch在程序执行时检视变量内容与函式呼叫的 关系。应用这个Compiler directive的注意事项有:$D-时, $L+不会有作用。使用$L+, 只会加大.DCU的档案大小, 对.EXE的大小与执行速度并没有影响。$H+ 长字符串宣告Delphi 2.0之后, 字符串多了一个更为好用的长字符串, 不仅没有数据长度255的限制与C语言惯用的Null-terminated string兼容性也大为提高。使用$H时的注意事项有:$H+的编译情形下, 以string定义的字符串变量都是长字符串, 请注意, 字符串是否为长字符串是在字符串定义时决定的, 例如:procedure TForm1.Button1Click(Sender: TObject);$H-vars: string;begin$H+s := 测试一下长字符串;Windows.MessageBox(0, pchar(s), 讯息, 64);end;由于var前$H-的缘故, 虽然在begin后我们立即设定为$H+, 但s仍然是一个短 字符串, 所以, 自然不能像是长字符串一样, 以pchar强制型别转换后当作Null-terminated字符串使用。承上, 不管程序是$H+或$H-, 只要字符串是以长字符串方式定义的, 即使begin.end;中改成$H-, 该字符串的操作仍然具有长字符串的特性。因此, 由于VCL中的字符串都是长字符串, 即使我们的程序是$H-, 仍然可以拿它们 当长字符串来使用。不论$H的状态如何, 以AnsiString定义的一定是长字符串; 以stringn或ShortString定义的一定是短字符串。$M 16386, 1048576 内存配置大小要改变唯迭(Stack)内存配置大小时, 我们可以有以下两种选择:使用$MINSTACKSIZE数字, $MAXSTACKSIZE数字, 分别指定最小.最大的Stack大小.或者使用$M min, max, 同时指定最小与最大的值。使用这些Compiler directive时的注意事项有:写在.DPR中才有效果。堆栈的最小数字必须介于1024至21474835647之间。堆栈的最大数字必须介于$MINSTACKSIZE至21474835647之间。当内存不足而无法满足最小的堆栈大小时, Windows会在启动这程序时提出错 误报告。当程序要求的内存超过$MINSTACKSIZE的大小时, 将举发EStackOverflow例外 。$Z1 最小列举大小这个Compiler directive将影响储存列举型态时最小所需的byte数值。如果宣告 列举型态时, 数值不大于256, 而且也在系统预设的$Z1时, 这个列举型态只占 用一个byte储存的。$Z2时, 以两个byte储存, $Z4时, 以四个byte储存。因 为C语言通常以WORD或DWORD储存列举型态, 如果您的程序需要与C、C+沟通时,$Z2$Z4就很管用了$Z+, 与$Z-分别对应到$Z1和$Z4。$P+ 开放字符串参数在程序与函数宣告时, 其中的字符串自变量, 在$P+时表示是Open string; $P-时 , 只是一般的字符串变量而已。这个Compiler directive只在$H-时有作用。$O+ 最佳化开关建议您维持$O+的系统默认值。开启这个Compiler directive, Delphi会自动进行最佳化处理, 程序可以因此跑得快一些, 您可以放心的打开这个编译开关, Delphi不会进行不安全的最佳化而使您的程序执行时发生错误。$Q- 满溢检查, $R- 范围检查$Q与$R是一组搭配使用的Compiler directive, 它们将检查数值或数组的操作是否在安全的边界中, $Q会检查整数运算(如+, -, Abs, Sqr, Pred, Succ等 ), 而$R则检查字符串与数组的存取是否超出合理边界范围等问题。使用这两个Compiler directives会因为这些检查动作而降低程序执行的速度, 通 常我们会在除错时开启这两个编译开关。$U- Pentiu
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 比亚迪电工知识培训课件
- 仓库文员工作总结
- 音乐教学课件制作学什么好
- 学前教育概述试题及答案
- 第一单元 有机化学的发展与应用教学设计-2025-2026学年高中化学苏教版2019选择性必修3-苏教版2019
- 2025年冰雪旅游项目投资可行性评估与可持续发展策略报告
- 七夕节优惠活动方案
- 2025年智能微电网在新能源社区中的应用案例分析及启示报告
- 云南省楚雄市古城二中2025年生物高三上期末复习检测模拟试题
- Unit 7 Whats the highest mountain in the world Section A 3a-3c 说课稿 2025-2026学年人教版英语八年级下册
- 2024年德州市第二人民医院招聘备案制工作人员笔试真题
- 护理沟通与服务课件
- 高低压配电施工设计方案
- 2025年辽宁省高考历史试卷及答案详解
- 企业种子管理制度
- 2025年村干部考试试题及答案
- 上海长租公寓市场现状和未来发展趋势分析
- 焊工安全操作技术规范
- 医疗安全不良事件培训课件
- 2025+CSCO结直肠癌诊疗指南解读
- 儿童体态课件
评论
0/150
提交评论