VarToDateAsString
调用 'oleaut32'sVarDateFromStr
来执行转换,如果你有中间的年份,它就会失败。
variants.VarToDateAsString
:
function VarToDateAsString(const V: TVarData): TDateTime;
var
...
begin
_VarToWStr(S, V);
LResult := VarDateFromStr(S, VAR_LOCALE_USER_DEFAULT, 0, Result);
...
在第二行放一个断点,你会看到Result
参数会颠倒日期和年份。您可以通过调用自己进行测试activex.VarDateFromStr
。
您可以绕过variants.VarToDateAsString
调用VarDateFromStr
并实现自己的函数,但不要考虑使用 from 的东西,sysutils
因为它也不支持中间格式的一年。依次sysutils.TryStrToDateTime
调用. 下面是整个函数:ScanDate
GetDateOrder
function GetDateOrder(const DateFormat: string): TDateOrder;
var
I: Integer;
begin
Result := doMDY;
I := 1;
while I <= Length(DateFormat) do
begin
case Chr(Ord(DateFormat[I]) and $DF) of
'E': Result := doYMD;
'Y': Result := doYMD;
'M': Result := doMDY;
'D': Result := doDMY;
else
Inc(I);
Continue;
end;
Exit;
end;
end;
如您所见,中间没有年份排序。
看起来你必须自己解析字符串:
uses
varutils;
function MyVarDateFromStr(const strIn: WideString; LCID: DWORD; dwFlags: Longint;
out dateOut: TDateTime): HRESULT; stdcall;
begin
// write your parser here and return a valid 'dateOut'
end;
procedure TForm1.Button1Click(Sender: TObject);
var
LDateTimeVar: Variant;
LDateTime: TDateTime;
begin
varutils.VarDateFromStr := MyVarDateFromStr; // replace the broken function
// Current date separator in OS settings is ';'
LDateTimeVar := '07;12;18';
LDateTime := VarToDateTime(LDateTimeVar);
// Expected LDateTime = '07;12;18',
// but will be LDateTime = '07;18;12'
ShowMessage(DateToStr(LDateTime));
end;