4

我正在尝试将动态对象移交给 Ironpython,但似乎 Ironpython 没有调用 TryInvokeMember。相反,它调用 TryGetMember 并给出无法调用结果的错误。

我已经尝试过 IronPython 2.7 和 2.6.10920

示例代码:

动态对象:

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 必须是一个列表,否则嵌套方法不起作用)

所以感谢杰夫。

当我这样设计动态时:

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 不知道被调用方法的名称。所以我将它存储到一个变量中。

4

2 回答 2

2

这是预期的行为。IronPython 使用TryGetMember后跟,TryInvoke因为 Python 语言没有调用成员的概念:Python 方法调用始终是属性查找后跟调用。

您收到的错误是因为您设置result为字符串(类型str)并且字符串不可调用。您必须设置result为另一个实现的动态对象TryInvoke或委托。

于 2013-06-14T15:07:19.710 回答
0

基于 Python 异常(str 不可调用),它看起来像 IronPython 第一次调用TryGetMember,并且由于该函数返回 true,它将使用该对象。

只有在他们成功的TryGetMember/TryInvokeMember情况下才应该返回。true例如,这意味着当且仅当实际上存在与参数匹配的成员时才TryGetMember应返回。truebinder

更新:在 C#Call(2)中总是一个方法调用(我想不出还有什么可能,如果我错了请告诉我),所以 C# 编译器将使用TryInvokeMethod. 然而,在 Python 中,任何对象如果有__call__方法都是可调用的,因此Call(2)可能意味着执行该方法Call或获取成员Call__call__在其上执行。

于 2013-06-14T12:04:56.360 回答