Why does Format crash when anything but “%s” is used with a Variant?

烈酒焚心 提交于 2019-12-22 03:11:49

问题


I'm working with the SysUtils.Format function and variant values, and I found that this function only works if the format string is %s. I checked the documentation about the Format function but there does not exist any reference to how variant values are treated.

Consider this simple application:

{$APPTYPE CONSOLE}

uses
  Variants,
  SysUtils;

procedure TestFormat;
var
  v : Variant;
begin
  v:=100;
  writeln(Format('The VarType of v is %s',[VarTypeAsText(VarType(v))]));
  writeln(Format('The value of v is %s',[v]));//ok

  v:='100';
  writeln(Format('The VarType of v is %s',[VarTypeAsText(VarType(v))]));
  writeln(Format('The value of v is %s',[v]));//ok

  v:=100;
  writeln(Format('The VarType of v is %s',[VarTypeAsText(VarType(v))]));
  writeln(Format('The value of v is %d',[v]));//raise a EConvertError exception EConvertError: Format '%d' invalid or incompatible with argument
end;


begin
  try
     TestFormat;
  except
    on E: Exception do
      Writeln(E.ClassName, ': ', E.Message);
  end;
  readln;
end.

Is this a bug or a simple limitation of this function?

I've checked this behavior in Delphi 5, Delphi 2007 and Delphi XE.


回答1:


It is a limitation of the function. In Delphi XE, the relevant part in SysUtils starts at line 10870, which looks like this:

@CvtVariant:
        CMP     CL,'S'
        JNE     @CvtError

This is called for any variant argument. The CL register have the type required by the format string for that particular argument, for anything different than 'S', the exception is raised.




回答2:


It's a limitation of the function. For a more feature-rich version of Format, try the WideFormat function from the JCL. (I'm its author.) It supports Variants of various types, Boolean, and TClass. It also accepts character-pointer types for the %p format, and Int64 and Variant values for index arguments.

Despite its extensions, it was removed from the JCL distribution about a year ago because its primary target was Delphi 5, which didn't provide a native WideString version of Format, and the JCL no longer supports Delphi 5. The last revision that included it was 3140.



来源:https://stackoverflow.com/questions/5542366/why-does-format-crash-when-anything-but-s-is-used-with-a-variant

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!