Delphi的Format函数大家都用得很多,第二个参数用着确实很方便。最近在数据库开发应用中需要自己创建一个带array of const参数的函数,对于常用的类型String,Integer,Pointer处理都没什么问题,但当用到Widestring类型时却出错,摸索了一上午,感觉获益良多。现在将问题、解决问题的思路、分析方法等一一道来,希望对诸君有所启发就达到了我写这篇文章的目的了!
///环境:Winxp + D7 ///进入D7,在默认的新建工程中增加一过程Test(m: Array ofconst); procedure TForm1.test(m: arrayofconst); var i, zero: Integer; s, t: String; c: Char; const sBoolean: Array [Boolean] of string = ('False', 'True'); begin s :=''; for i :=0to High(m) dowith m[i] do case VType of//写到这,按住Ctrl点击VType,打开System单元,将VType的枚举值贴到Case语句 vtInteger: (VInteger: Integer; VType: Byte); vtBoolean: (VBoolean: Boolean); vtChar: (VChar: Char); vtExtended: (VExtended: PExtended); vtString: (VString: PShortString); vtPointer: (VPointer: Pointer); vtPChar: (VPChar: PChar); vtObject: (VObject: TObject); vtClass: (VClass: TClass); vtWideChar: (VWideChar: WideChar); vtPWideChar: (VPWideChar: PWideChar); vtAnsiString: (VAnsiString: Pointer); vtCurrency: (VCurrency: PCurrency); vtVariant: (VVariant: PVariant); vtInterface: (VInterface: Pointer); vtWideString: (VWideString: Pointer); vtInt64: (VInt64: PInt64); end; Delete(s, 1, 1); Self.Caption := s; end; ///继续写,对各枚举值进行处理!这里作一下解释,Array ofconst正是由TVarRec类型组成的! ///请看Case of语句中的代码: vtInteger: s := s +';'+ IntToStr(VInteger); vtBoolean: s := s +';'+ sBoolean[VBoolean]; vtChar: s := s +';'+ VChar; vtExtended: s := s +';'+ FloatToStr(VExtended^); vtString: if Assigned(VString) thenbegin t := VString^; s := s +';'+ t; end; vtPointer: if Assigned(VPointer) then s := Format('%S; Pointer: $%X ',[s, Integer(VPointer)]); vtPChar: if Assigned(VPChar) thenbegin t := VPChar^; s := s +';'+ t; end; vtObject: if Assigned(VObject) then s := Format('%S; $%X ClassName: %S ',[s, Integer(@VObject), VObject.ClassName]); vtClass: if Assigned(VClass) then s := Format('%S; Class Reference $%X - ClassName: %S ',[s, Integer(VClass), VClass.ClassName]); vtWideChar: begin t := VWideChar; s := s +';'+ t; end; vtPWideChar: if Assigned(VPWideChar) thenbegin t := VPWideChar^; s := s +';'+ t; end; vtAnsiString: if Assigned(VAnsiString) thenbegin t := PChar(VAnsiString); s := s +';'+ t; end; vtCurrency: if Assigned(VCurrency) then s := s +';'+ FloatToStr(VCurrency^); vtVariant: if Assigned(VVariant) then s := s +'; This is variant '; vtInterface: if Assigned(VInterface) then s := Format('%S; Interface: $%X',[s, Integer(VInterface)]); vtWideString: if Assigned(VWideString) thenbegin t := PWideString(VWideString)^; s := s +';'+ t; end; vtInt64: if Assigned(VInt64) then s := s +';'+ IntToStr(VInt64^);
加上一按钮测试该函数
procedure TForm1.Button1Click(Sender: TObject); var ws: WideString; begin ws :='dda这是一个测试dfa'; test([self, 'sdf', 2.3324, ws, TForm]); end;
可以看到测试结果,变量ws的值没有显示出来,怎么办呢?
我们可以看到WideString类型的值是指针,我们就从这里着手,在事件中添加一句:
Button1.Caption := Format('$%X',[Integer(@ws)]);
此句的作用是显示出ws的地址
再在Test函数中也加上类似的语句,并注释掉无用的语句:
//t := PWideString(VWideString)^;
//s := s + ';' + t;
s := s + ';' + Format('$%X',[Integer(VWideString)]);
运行可看到二个地址不一样,说明Delphi对传入的参数数据作了复制
因此将其强制转换成PWidechar应该可以,增加一变量声明
w: WideString;
w := PWideString(VWideString)^;
s := s + ';' + w;
但运行结果却只显示一个字符,别沮丧,已经摸到门道了!
if Assigned(VWideString) then begin
t := '';
zero := 0;
p := VWideString;
repeat
c := char(p^);
inc(p);
if c = #0 then
inc(zero) else
begin
zero := 0;
t := t + c;
end;
until zero = 2;
s := s + ';' + t;
end;
if Assigned(VWideString) then begin
t := '';
p := VWideString;
repeat
t := t + widechar(p^);
inc(p);
until p^ = 0;
s := s + ';' + t;
end;
可以看到核心代码已经很精练了,运行已经显示正常,汉字也无乱码了!至此我们似乎是大功告成了,但静下来想想,Delphi支持WideString到String的转换,它也应该有这样的处理代码。
而且在循环中t := t + widechar(p^);语句处下一断点,运行到断点处,再打开CPU窗口,看到看似简洁的代码,单此一句,编译器都要给它加上一大堆处理代码。找到系统的字符串处理函数很有必要,经过在System.pas单元中搜索WideString,找到函数:procedure WideCharToStrVar(Source: PWideChar; var Dest: string);
呵呵,这正是我们要的!!!
现在循环语句及P变量都可删除了,代码我就省略了。