IronPython调用TryGetMember而不是TryInvokeMember
本文关键字:TryInvokeMember 调用 TryGetMember IronPython | 更新日期: 2023-09-27 18:18:18
我试图将一个动态对象移交给Ironpython,但似乎Ironpython没有调用TryInvokeMember。相反,它调用TryGetMember并给出一个错误,它不能调用结果。
我已经在IronPython 2.7和2.6.10920中尝试过了
ExampleCode:
DynamicObject:
class ExampleDynamicObject: DynamicObject {
public override bool TryGetMember(GetMemberBinder binder,
out object result) {
result = "TryGetMember";
return true;
}
public override bool TryInvokeMember(InvokeMemberBinder binder,
object[] args,
out object result) {
result = "TryInvokeMember";
return true;
}
}
调用Mathode static void Main(string[] args) {
dynamic example = new ExampleDynamicObject();
var program = @"test = example.Call2(2)";
var engine = Python.CreateEngine();
var scope = engine.CreateScope();
scope.SetVariable("example", example);
var source = engine.CreateScriptSourceFromString(program,
SourceCodeKind.Statements);
source.Execute(scope);
Console.ReadKey();
}
调用TryGetMember方法,然后抛出Microsoft.Scripting.ArgumentTypeException "str is not callable"
当你编写类似'test'(1)
这样的代码时抛出。所以Python似乎没有意识到这是一个函数调用,而只是调用了一个属性。
但是当我尝试从c#中调用它时
Console.WriteLine(example.Call);
Console.WriteLine("----------------------------");
Console.WriteLine(example.Call(1));
TryGetMember
-------------------
TryInvokeMember
有没有人建议如何解决这个问题?
解决方案:(编辑:calledMethodeName必须是一个列表,否则嵌套的方法将无法工作)
感谢Jeff。
当我这样设计动态时:
List<string> calledMethodeNames = new List<string>();
public override bool TryGetMember(GetMemberBinder binder,
out object result)
{
calledMethodeNames.Add(binder.Name);
result = this;
return true;
}
public override bool TryInvoke(InvokeBinder binder, object[] args, out object result)
{
//calledMethodeNames last Element has stored the Name of the called methode (remeber to remove it)
result = "TryInvoke";
return true;
}
一切正常。
线索是返回对象本身作为成员,然后对象是可调用的并且python调用TryInvoke(而不是TryInvokeMember)
但是TryInvoke被调用,因为在返回对象之后,对象本身被调用。所以InvokeBinder不知道被调用方法的名字。所以我把它存储到一个变量中
这是预期的行为。IronPython使用TryGetMember
后跟TryInvoke
,因为Python语言没有调用成员的概念:Python方法调用总是属性查找后跟调用。
你得到的错误是因为你将result
设置为字符串(类型str
),字符串不可调用。您必须将result
设置为实现TryInvoke
的另一个动态对象或委托。
基于Python异常(str不可调用),看起来IronPython首先调用TryGetMember
,并且由于该函数返回true,它将使用该对象。
TryGetMember/TryInvokeMember
应该只在它们成功时返回true
。例如,这意味着当且仅当存在与binder
形参匹配的成员时,TryGetMember
应该返回true
。
更新:在c# Call(2)
总是一个方法调用(我不能想到任何其他可能的,请让我知道,如果我错了),所以c#编译器将使用TryInvokeMethod
。然而,在Python中,如果任何对象具有__call__
方法,则都是可调用的,因此Call(2)
可以表示执行方法Call
或获取成员Call
并对其执行__call__
。