Delphi是一種強類型轉換的語言。在VC中,賦值符用″=″,例如x=1;到了Delphi賦值符就變成了″:=″,例如x:=1。 從賦值時用符號″:=″而不用″=″,就隱約可見Delphi對類型匹配要求之嚴,即賦值符右邊的類型一定要和左邊一致。
用慣了VB或VC的程序員,初用Delphi,稍不留神,就會出現類型不匹配的錯誤。對初學者而言,類型轉換也是學習Delphi的重點和難點,為此本文特對Delphi的類型轉換做一總結,以供讀者參考。
一、數的類型轉換
把表達式的類型從一種類型轉化為另一種類型,結果值是把原始值截斷或擴展,符號位保持不變。例如:
數的類型轉換
舉例
字符轉換為整數
Integer('A')
整數轉換為字符
Char(48)
整數轉換為1個字節的邏輯型
Boolean(0)
整數轉換為2個字節的邏輯型
WordBool(0)
整數轉換為4個字節的邏輯型
LongBool(0)
整數轉換為10進制pascal型字符串
caption:=intToStr(15)
整數轉換為16進制pascal型4位字符串
caption:=intToHex(15,4)
地址轉換為長整型數
Longint(@Buffer)
二、數的“分開”與“合成”
取32位longint型數的高16位數為
hiword(longint-var)
低16位數為
loword(longint-var)
取16位數的 高8位數
hibyte(integer_var)
低8位數為
lobyte(integer_var)
取32位地址的段選擇符和偏移量 段選擇符(高16位地址)為
selectorof(p)
偏移量(低16位地址)為
offsetof(p)
段選擇符和偏移量合成為指針
Ptr(SEG, OFS: Word)相當於C語言的宏MK-FP(SEG,OFS)
例如在Windows中,Task DataBase結構0FAh偏移處包含'TD'標識,我們可以容易地編寫如下代碼觀察到這個位於Windows內部的未公開的秘密:
{函數ptr(seg,ofs)的用法,相當於C語言的MK-FP(seg,ofs)}
var p:pbyte;ch:char;
p:=ptr(getcurrentTask,$FA);
ch:=char(p^); {結果為ch='T'}
p:=ptr(getcurrentTask,$FA+1);
ch:=char(p^); {結果為ch='D'}
三、字符串string 字符數組與指向字符串的指針pchar的區別與聯系
這3者的基本概念相同,但有一些非常細微的差別,在編程時稍不注意就會出錯,需高度重視。
1、使用指向字符串的指針,如果不是以0結尾,運行時就會出現錯誤。為了避免這種錯誤,需要在字符串結尾人工加入0 即char(0),或用strpcopy函數在字符串結尾自動加0。
例1: 指向字符串的指針,如果不是以0結尾,運行時會出現錯誤:
{s[0]=3 s[1]='n' s[2]='e' s[3]='w'}
var
s:string;
p:pchar;
begin
s:='new';
label1.caption:=s; {new}
label2.caption:=intTostr(integer(s[0]));{3是字符串的長度}
p:=@s[1];{不是以0結尾,莫用pchar型指針}
label3.caption:=strpas(p); {運行時出現錯誤}
end;
例2:在字符串結尾人工加入0即char(0),可使用指向字符串的指針。
{s[0]=4 s[1]='n' s[2]='e' s[3]='w' s[4]=0;}
{p-->'new'}
var
s:string;
p:pchar;
begin
p:=@s[1];
s:='new'+char(0); {以0結尾,可用pchar型指針}
label1.caption:=strpas(p); {new}
label2.caption:=s; {new}
label3.caption:=intTostr(integer(s[0])); {4是字符串長度 end;
例3: 用strpcopy函數賦值會在字符串s結尾自動加0。
{s[0]=4 s[1]='n' s[2]='e' s[3]='w' s[4]=0;}
{p-->'new'}
var
s:string;
p:pchar;
begin
p:=@s[1];
strpcopy(p,'new');{strpcopy函數在字符串結尾自動加0}
label1.caption:=strpas(p);{new}
label2.caption:=s;{new}
label3.caption:=intTostr(integer(s[0]));{4}
end;
2、下標為0的字符串標識符存放的是字符串長度,字符型數組基本相當於字符串,但不能存放字符串長度。字符串可以用s:='a string'的形式賦值,但是字符型數組a[ ]不可直接用a:='array'的形式賦值,用此種形式會出現類型不匹配錯誤,可選用strpcopy函數賦值。
例4: 字符型數組s[ ]相當於字符串,但沒有存放字符串長度的位置。
{s[1]='n' s[2]='e' s[3]='w' s[4]=0;}
{p-->'new'}
var
s:array[1..10] of char;
p:pchar;
begin
{s:='new'+char(0); error}{錯誤}
p:=@s[1];
{p:=@s; is not correct}
strpcopy(p,'new');
label1.caption:=strpas(p);{new}
label2.caption:=s;{new}
{label3.caption:=intTostr(integer(s[0]));}{不會出現4, 下
超出錯誤}
end;
例5:下標從0開始的字符數組s,s相當於@s[0]。
{ s[0]='n' s[1]='e' s[2]='w' s[3]=0;}{p-->'new'}
var
s:array[1..10] of char;
p:pchar;
begin
{s:='new'+char(0); error}{錯誤}
p:=s;
{p:=@s[0] is also correct}
strpcopy(p,'new');
label1.caption:=strpas(p);{new}
label2.caption:=s;{new}
label3.caption:=s[0];{n}
end;
3、下標從0開始和從1開始的字符數組地址的表示方法也有細微不同:
例6:下標從1開始的數組a 與下標從0開始的數組b 的比較。
var
a:array[1..10]of char;
b:array[0..10]of char;
{a:='1..10';}{type mismatch}
{b:='0..10';}{type mismatch}
begin
strpcopy( b, 'from 0 to 10'); {正確 因為b即是@b[0] }
strpcopy(@b[0], 'from 0 to 10'); {正確 與上個表達式
果相同}
strpcopy(@a[1], 'from 1 to 10'); {正確 }
strpcopy( a, 'from 1 to 10'); {類型匹配錯誤 因為a即是a[0]}
end;