RRU*_*RUZ 2 delphi rtti delphi-xe
我需要确定a TRttiMethod是否是一个函数
到目前为止,我写了这个函数
Function IsFunction(QualifiedName,MethodName:string):Boolean;
Var
ctx : TRttiContext;
lType : TRttiType;
lMethod : TRttiMethod;
Begin
result:=false;
ctx := TRttiContext.Create;
lType:=ctx.FindType(QualifiedName);
if Assigned(lType) then
begin
lMethod:=lType.GetMethod(MethodName);
if Assigned(lMethod) then
Result:=(lMethod.ReturnType<>nil); //in this line the exception is raised
end;
End;
Run Code Online (Sandbox Code Playgroud)
但是Insufficient RTTI available to support this operation.当我测试以下时,此函数失败并出现此异常
IsFunction('SysUtils.Exception','CreateFmt')
Run Code Online (Sandbox Code Playgroud)
这些类和方法也失败了
SysUtils.Exception -> CreateFmt
SysUtils.Exception -> CreateResFmt
SysUtils.Exception -> CreateResFmt
SysUtils.Exception -> CreateFmtHelp
SysUtils.Exception -> CreateResFmtHelp
SysUtils.Exception -> CreateResFmtHelp
SysUtils.TEncoding -> Convert
SysUtils.TEncoding -> Convert
SysUtils.TEncoding -> GetBufferEncoding
SysUtils.TEncoding -> GetBufferEncoding
SysUtils.TEncoding -> GetByteCount
SysUtils.TEncoding -> GetByteCount
SysUtils.TEncoding -> GetByteCount
SysUtils.TEncoding -> GetByteCount
SysUtils.TEncoding -> GetBytes
SysUtils.TEncoding -> GetBytes
SysUtils.TEncoding -> GetBytes
SysUtils.TEncoding -> GetBytes
SysUtils.TEncoding -> GetBytes
SysUtils.TEncoding -> GetCharCount
SysUtils.TEncoding -> GetCharCount
SysUtils.TEncoding -> GetChars
SysUtils.TEncoding -> GetChars
SysUtils.TEncoding -> GetChars
SysUtils.TEncoding -> GetPreamble
SysUtils.TEncoding -> GetString
SysUtils.TEncoding -> GetString
SysUtils.TMBCSEncoding -> GetPreamble
SysUtils.TUTF8Encoding -> GetPreamble
SysUtils.TUnicodeEncoding -> GetPreamble
SysUtils.TBigEndianUnicodeEncoding -> GetPreamble
Rtti.TRttiMethod -> Invoke
Rtti.TRttiMethod -> Invoke
Rtti.TRttiMethod -> Invoke
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> AddRange
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> AddRange
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> AddRange
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> InsertRange
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> InsertRange
Generics.Collections.TList<Rtti.TMethodImplementation.TParamLoc> -> InsertRange
Generics.Collections.TArray -> Sort
Generics.Collections.TArray -> Sort
Generics.Collections.TArray -> Sort
Generics.Collections.TArray -> BinarySearch
Generics.Collections.TArray -> BinarySearch
Generics.Collections.TArray -> BinarySearch
Run Code Online (Sandbox Code Playgroud)
我写了这个小应用程序来检查这种行为
program ProjectTest;
{$APPTYPE CONSOLE}
uses
Rtti,
SysUtils;
Function IsFunction(QualifiedName,MethodName:string):Boolean;
Var
ctx : TRttiContext;
lType : TRttiType;
lMethod : TRttiMethod;
Begin
result:=false;
ctx := TRttiContext.Create;
lType:=ctx.FindType(QualifiedName);
if Assigned(lType) then
begin
lMethod:=lType.GetMethod(MethodName);
try
if Assigned(lMethod) then
Result:=(lMethod.ReturnType<>nil);
except on e : exception do
Writeln(Format('%s %s -> %s',[e.Message,QualifiedName,MethodName]));
end;
end;
End;
var
ctx : TRttiContext;
lType : TRttiType;
lMethod : TRttiMethod;
begin
try
ctx := TRttiContext.Create;
for lType in ctx.GetTypes do
for lMethod in lType.GetDeclaredMethods do
IsFunction(lType.QualifiedName,lMethod.Name);
Readln;
except
on E: Exception do
Writeln(E.ClassName, ': ', E.Message);
end;
end.
Run Code Online (Sandbox Code Playgroud)
这是确定TRttiMethod一个函数是否正确的正确方法?
UPDATE
由于@Barry建议,我重写了函数以避免异常,但是这并没有解决TRttiMethod由于RTTI的当前限制如何确定a 是否是函数的问题.
function IsFunction(lType : TRttiType;MethodName:string):Boolean;
Var
ctx : TRttiContext;
lMethod : TRttiMethod;
Begin
result:=false;
if Assigned(lType) then
begin
lMethod:=lType.GetMethod(MethodName);
if Assigned(lMethod) then
if lMethod.HasExtendedInfo then
Result:= (lMethod.MethodKind in [mkFunction,mkClassFunction]) //you can be 100 % sure, wich this is a function or not
else // sorry but the RTTI does not provide this information when the TRttiMethod contains parameters or an resultype that are not supported by the RTTI
Result:=False;
end;
End;
Run Code Online (Sandbox Code Playgroud)
您可以检查TRttiMethod.HasExtendedInfo以避免异常.该类会尝试访问仅在HasExtendedInfo返回时数据可用的属性True.
你可以考虑检查MethodKind属性,看它是否mkFunction或者mkClassFunction根据需要,等等.如果是,则MethodKind返回.mkProcedureHasExtendedInfoFalse