C#/ IronPython与共享C#类库互操作

我正在尝试使用IronPython作为C#GUI和一些C#库之间的中介,以便它可以在编译后编写脚本。

我有一个类库DLL,由GUI和python使用,是这样的:

namespace MyLib { public class MyClass { public string Name { get; set; } public MyClass(string name) { this.Name = name; } } } 

IronPython代码如下:

 import clr clr.AddReferenceToFile(r"MyLib.dll") from MyLib import MyClass ReturnObject = MyClass("Test") 

然后,在C#中我将其称为如下:

 ScriptEngine engine = Python.CreateEngine(); ScriptScope scope = null; scope = engine.CreateScope(); ScriptSource source = engine.CreateScriptSourceFromFile("Script.py"); source.Execute(scope); MyClass mc = scope.GetVariable("ReturnObject ") 

当我调用最后一段代码时,source.Execute(scope)运行成功返回,但是当我尝试GetVariable调用时,它抛出以下exception

 Microsoft.Scripting.ArgumentTypeException: expected MyClass , got MyClass 

所以,你可以看到类名完全相同,但由于某种原因,它认为它们是不同的。

DLL与.py文件位于不同的目录中(我只是没有写出所有路径设置的东西),可能是因为IronPython的解释器存在问题,因为它看起来不同,因为它是某种方式看到他们处于不同的背景或范围?

此错误表示您的程序集已加载到多个CLR加载程序上下文中。 您可以切换到clr.AddReference,也可以从C#加载程序集,而不是使用clr.AddReferenceToFile添加引用。 对于前者,您需要确保程序集在.NET可以正常加载它的位置(GAC或进程的应用程序库)中可用。 对于后者你可以这样做:

 engine.Runtime.LoadAssembly(typeof(MyClass).Assembly); 

来自您的C#主机代码。 就个人而言,我更喜欢这个第二个解决方案,因为它不仅可以帮助您节省用户从Python进行clr.AddRef调用。

您可以尝试在调试器下运行程序并在GetVariable调用之前中断执行。 转到“模块”窗口,查看是否加载了两个版本的C#类库DLL。 如果是这样,那就是解释。

如果这是问题,那么解决方案是确保C#和Python世界在类型上达成一致。 一种解决方案是将所有内容放在同一目录中。 另一种可能性是通过使用ScriptScope类(我认为)上的属性来设置对C#中类库的引用,以设置可供Python代码使用的类库程序集的程序集引用。 我没有立即可用于测试的混合C#/ IronPython项目,但我记得看到了这个function。