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而不是TryInvokeMember

这是预期的行为。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__