我学Delphi心得及笔记----类型、变量及常量(第二讲)

来源:互联网 发布:淘宝网上买东西可靠吗 编辑:程序博客网 时间:2024/06/06 01:46

======================================================
注:本文源代码点此下载
======================================================

1、变量

pascal 变量在使用前必须声明,声明变量时必须指定一种数据类型。下面是变量声明的例子:

varvalue: integer;iscorrect: boolean;a, b: char; 申明变量以var为关键字,格式为:var 变量名:类型;

2、常量

对于在程序运行期间保持不变的值,pascal 允许通过常量来声明。声明常量不必特定数据类型,但需要赋一个初值。编译器会根据所赋初值自动选用合适的数据类型。例如: const

thousand = 1000;

pi = 3.14;

authorname = 'marco cantù';常量及为在编辑与运行期间不可对其值进行改变;

delphi 根据常量的值来决定它的数据类型。上例中的thousand 变量,delphi会选用smallint数据类型 (短整型--能容纳thousand变量的最小整数类型)。如果你想告诉delphi 采用特定的类型,你可在声明中加入类型名,方法如下:

constthousand: integer = 1000;

对于声名的常量,编译器有两种编译选择:第一种为常量分配内存,并把常量的值放入内存;第二种在常量每次使用时复制常量值。第二种方法比较适合简单常量。

3、资源串常量

当定义字符串常量时,你可这样写:

constauthorname = 'marco cantù';

从delphi 3 开始,你可以用另一种方式写:

resourcestringauthorname = 'marco cantù';

上面两个语句都定义了一个常量,也就是定义了一个在程序运行期间保持不变的值,但两者的实现过程却不同,用resourcestring 指令定义的字符串变量将被保存到程序资源的字符串表中。从例子resstr你可了解资源串的实际作用,例子中设置了一个按钮, 相应代码如下:

resourcestringauthorname = 'marco cantù';bookname = 'essential pascal';procedure tform1.button1click(sender: tobject); beginshowmessage (bookname + #13 + authorname); end;

以上代码中的两个字符串将分两行输出显示,因为字符串被分行符 #13 隔开。

有趣的是,当你用资源编辑器打开执行文件时,你会在程序资源中看到你所定义的字符串。这意味着字符串并没有进入编译代码,而是保存在执行文件 (exe文件) 的一个单独区域。

注意:简而言之,采用资源的好处一方面可让windows 来完成有效的内存处理,另一方面不用更改源代码就可实现程序的本地化 (把字符串翻译成不同的语言)。

4、数据类型

pascal中有多种预定义的数据类型,它们可分为三大类:有序数据类型,实数类型和字符串类型。下面我们先讨论有序类型和实数类型,字符串类型放在以后讨论。同时这 一节还将介绍几种delphi 库中定义的类型 (不是编译器预定义的类型),这些类型也可看作是预定义的类型。

delphi 还包括一种无类型的可变数据类型,称作variant,在本书的第十讲将讨论这一类型。variant是一种无需类型检测的数据类型,它在delphi 2 中引入,用于处理ole automation(ole 自动化)。

5、有序类型

有序类型是建立在概念“顺序”或“序列”基础上的数据类型。你不仅可比较两个有序值的大小,而且可以求取给定有序值的前驱及后继,或者计算它们的最大或最小值。

三种最重要的预定义有序类型是整数类型、布尔类型和字符类型(integer,boolean,char)。各种类型根据其内部表示和取值范围不同又可进一步细分。表3.1列出了表示数字的有序数据类型。

表 3.1: 表示数字的有序数据类型

大小有符号值域无符号值域

8 bits

shortint

-128 to 127

byte

0 to 255

16 bits

smallint

-32768 to 32767

word

0 to 65,535

32 bits

longint

-2,147,483,648 to 2,147,483,647

longword (从 delphi 4)

0 to 4,294,967,295

64 bits

int64

16/32 bits

integer

cardinal

从表中可看到,不同数据类型与不同的数据表示法相对应,这要取决于数据值的数位和符号位。有符号类型的数值可正可负,但取值范围较小,因为符号位占一个数位。下一节在例range中说明了每种类型的实际取值范围。

表中最后一组类型标志着16/32,它表明其数值表示方法在16位和32位delphi中不同,该组的integer及cardinal 类型比较常用,因为它们与cpu内部的数字表示法相对应。

delphi 4中的整数类型

在 delphi 3中,cardinal类型所表示的32位无符号值实际占31位,取值最高为20亿。delphi4新增了一种无符号数字类型--longword,它是真正的32位值,取值最高达40亿。现在cardinal类型已成了longword类型的别名,只是longword能容纳大于20亿的无符号数,而且它的数值表示法与cpu内部数值表示法一致。

delphi 4 中新增的另一个数据类型是int64 类型,这一类型能表示长达18个数字的整数。系统中的有序类型例程(如high 和low)、数字例程(如inc 和 dec)及字符串转换例程(如inttostr)都支持这一新类型。反过来,有两个新增的专用函数strtoint64 和 strtoint64def支持从字符串向数字的转换。

布尔类型

布尔值不同于布尔类型,平时很少用到。bytebool、 wordbool 和longbool这三种布尔类型的布尔值比较特殊,只在windows api 函数中才用到它们。

在delphi 3 中,为了与visual basic 和 ole automation兼容,修改了bytebool、 wordbool 和longbool的布尔值,将true值设置为1,false值仍为0;boolean类型布尔值保持不变(true为1,false为0)。如果在delphi 2代码中使用了布尔值显式类型转换 ,那么在以后的delphi中可能会出错。

字符类型

字符有两种不同的表示法:: ansichar 和 widechar。第一种类型代表 8位的字符,与windows一直沿用的ansi(美国国家标准协会)字符集相应;第二种类型代表 16 位的字符,与windowsnt、windows 95 和 98支持的双字节字符(unicode)相应。在delphi 3 中,char类型字符与ansichar一致。切记,不管在什么环境,前 256 个unicode 字符与ansi 字符是完全一致的。

常量字符可用代表它们的符号表示,如‘k’,也可用数字符号表示,如 #78。后者还可用chr函数表示为 chr(78),用ord函数可作相反的转换ord(k)。

一般来说,对字母、数字或符号,用代表它们的符号来表示较好;而涉及到特殊字符时用数字符号较好。下面列出了常用的特殊字符:

#9 跳格 (tab 键)

#10 换行

#13 回车 (enter 键)

有序类型系统例程

pascal 语言和delphi system 单元中定义了一系列有序类型操作例程,见表 3.2。c++ 程序员会注意到其中的inc 例程,它可与 ++ 和 += 运算符对应(dec 例程也同样)。

表 3.2: 有序类型系统例程

例程作用

dec

将例程中的参数值递减1或一个特定的值,其中特定值可在第二个可选参数中定义

inc

将例程中的参数值增加1或一个特定的值

odd

如果参数为奇数返回真

pred

根据参数在其数据类型定义中的序列,返回参数值的前驱值

succ

返回参数值的后继值

ord

返回参数值在其数据类型值集合中的序号

low

返回参数对应的有序数据类型的最小取值

high

返回参数对应的有序数据类型的最大取值

注意,当有些例程用于常量时,编译器会自动用计算值替代例程。例如你调用high(x) ,设定x为一个整数,那么编译器会用整数类型中最大的可能值代替这个表达式。

实数类型

实数类型代表不同格式的浮点数。single类型占的字节数最小,为4个字节;其次是double 浮点类型,占8个字节;extended 浮点类型,占10个字节。这些不同精度的浮点数据类型都与ieee( 电气和电子工程师协会)标准的浮点数表示法一致,并且 cpu数字协处理器直接支持这些类型,处理速度也最快。

real 类型在delphi 2 和 delphi 3 中的定义与 16 位版本一样,都占 6 个字节。不过borland公司一直不提倡使用这种类型,而建议用single、 double、 extended 类型代替。这是由于 real 这种 6 字节的旧格式既不受 intel cpu 的支持,又没有列在官方的ieee 实型中。为了完全解决这一问题,delphi 4 不得不修改 real 类型的定义,将其改成标准的 8 字节浮点型, 由此引起了兼容性问题,不过如果有必要,你可以采用下面编译指令克服兼容性问题,恢复delphi 2 和 delphi 3 的real 类型定义:

{$realcompatibility on}

另外还有两种奇怪的数据类型:comp 类型和currency 类型,comp 类型用 8 个字节描述非常大的整数(这种类型可支持带有 18 位小数的数字);currency 类型 (16 位版的delphi不支持该类型) 表示一个有四位小数位的值,它的小数位长度是固定的,同comp 类型一样也占 8 个字节。正如名字所示,currency 数据类型是为了操作很精确的四位小数货币数值才添加的。

对实型数据,我们没办法编一个类似range的程序,因为high 、low及 ord函数不能用于实型值。理论上说实型类型代表一个无限的数字集合;有序类型代表一个有限的数字集合。

注意:让我进一步把上述问题解释一下。对于整数 23,你能确定23后面的数是什么 ,因为整型数是有限的,它们有确定的值域范围及排列顺序。而浮点数即使在一个很小的值域范围内也无限、无序。 事实上,在 23 和24 之间有多少值? 哪个值是 23.46 后面的值? 23.47 还是 23.461,或者 23.4601? 这是很难说清的。

日期和时间

delphi 也用实型数表示日期和时间数据。但为了更准确起见,delphi 特别定义了tdatetime 数据类型,这是一个浮点类型,因为这个类型必须足够宽,使变量能容纳年、月、日、时、分和秒、甚至毫秒。日期值按天计数,从1899-12-30开始,放在tdatetime 类型的整数部分;时间值则位于十进制数的小数部分。

tdatetime 不是编译器可直接识别的预定义类型,它在system单元定义:

type

tdatetime = type double;

使用tdatetime 类型很简单,因为delphi 为该类型定义了一系列操作函数,表3.3列出了这些函数。

表3.3: tdatetime类型系统例程

例程作用

now

返回当前日期及时间

date

返回当前日期

time

返回当前时间

datetimetostr

按缺省格式将日期和时间值转换为字符串;特定格式转换可用 formatdatetime函数

datetimetostring

按缺省格式将日期和时间值拷贝到字符串缓冲区

datetostr

将tdatetime值的日期部分转为字符串

timetostr

将tdatetime值的时间部分转为字符串

formatdatetime

按特定格式将日期和时间值转换为字符串

strtodatetime

将带有日期和时间信息的字符串转换为tdatetime类型值,如串有误将引发一个异常

strtodate

将带有日期信息的字符串转换为tdatetime类型格式

strtotime

将带有时间信息的字符串转换为tdatetime类型格式

dayofweek

根据传递的日期参数计算该日期是一星期中的第几天

decodedate

根据日期值返回年、月、日值

decodetime

根据时间值返回时、分、秒、毫秒值

encodedate

组合年、月、日值为tdatetime类型值

encodetime

组合时、分、秒、毫秒值为tdatetime类型值

到目前为止,我们所看到的预定义数据类 型都是pascal 语言自身定义的类型。 delphi中还包含windows系统定义的数据类型,这些数据类型不是pascal语言的组成部分,而是windows 库的一部分。windows类型包括新增的缺省类型(例如dword 或uint)、各种记录(或结构)类型及指针类型等。

windows 定义的数据类型中,最重要的类型是句柄(handle),第九讲中将讨论这一类型。

类型映射及类型转换

正如所知,你不能把一个变量赋给另一个不同类型的变量,如果你需要这么做,有两种方法供选择。第一种方法是采用类型映射(typecasting),它使用一个带有目标数据类型名的函数符号:

varn: integer;c: char;b: boolean; beginn := integer ('x');c := char (n);b := boolean (0);

你可以在字节长度相同的数据类型之间进行类型映射。在有序类型之间或实型数据之间进行类型映射通常是安全的,指针类型及对象之间也可以进行类型映射 ,只要你明白自己在做什么。

然而,一般来说类型映射是一种较危险的编程技术,因为它允许你访问一个似是而非的值,该值好象是其它值的替身。由于数据类型的内部表示法之间通常互相不匹配,所以当遇到错误时会难以追踪,为此你应尽量避免使用类型映射。

第二种方法是使用类型转换例程。表3.4中总结了各种类型转换例程。其中有些例程所涉及的数据类型将在下一节中讨论。 注意表中没有包括特殊类型(如tdatetime 和variant)的转换例程,也没包括用于格式化处理的特殊例程,如format 和formatfloat 例程。

表3.4:类型转换系统例程

例程作用

chr

将一个有序数据转换为一个ansi字符

ord

将一个有序类型值转换为它的序号

round

转换一个实型值为四舍五入后的整型值

trunc

转换一个实型值为小数截断后的整型值

int

返回浮点数的整数部分

inttostr

将数值转换为字符串

inttohex

将数值转换为十六进制数字符串

strtoint

将字符串转换为一个整型数,如字符串不是一个合法的整型将引发异常

strtointdef

将字符串转换为一个整数,如字符串不合法返回一个缺省值

val

将字符串转换为一个数字(传统turbo pascal例程用于向后兼容)

str

将数字转换为格式化字符串(传统turbo pascal例程用于向后兼容)

strpas

将零终止字符串转换为pascal类型字符串,在32位delphi中这种类型转换是自动进行的

strpcopy

拷贝一个pascal类型字符串到一个零终止字符串, 在32位delphi中这种类型转换是自动进行的

strplcopy

拷贝pascal类型字符串的一部分到一个零终止字符串

floattodecimal

将一个浮点数转换为包含指数、数字及符号的十进制浮点记录类型

floattostr

将浮点值转换为缺省格式的字符串

floattostrf

将浮点值转换为特定格式的字符串

floattotext

使用特定格式,将一个浮点值拷贝到一个字符串缓冲区

floattotextfmt

同上面例程,使用特定格式,将一个浮点值拷贝到一个字符串缓冲区

strtofloat

将一个pascal字符串转换为浮点数

texttofloat

将一个零终止字符串转换为浮点数

注意: 在最近版本的delphi pascal 编译器中,round 函数是以 cpu 的 fpu (浮点部件) 处理器为基础的。这种处理器采用了所谓的"银行家舍入法",即对中间值 (如 5.5、6.5) 实施round函数时,处理器根据小数点前数字的奇、偶性来确定舍入与否,如 5.5round 结果为 6,而 6.5 round 结果也为6, 因为 6 是偶数。特定的windows 类型


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
原创粉丝点击