我有一个框架,允许我使用键盘访问项目中对象的状态和方法。它严重依赖ImpromptuInterface,这是一个伟大,快速,灵活的东西。
例如,我使用Impromptu.InvokeMember(myObject, methodName, castParameters)
调用方法。它对公共和私人成员都很有用,但是当我尝试调用myObject
基类的私有成员时,我得到Microsoft.CSharp.RuntimeBinder.RuntimeBinderException: 'MyType.MyMethod(Something)' is inaccessible due to its protection level
。
揭示问题的最简单的代码:
public class MainClass
{
public static void Main(string[] args)
{
var type = new PublicType();
var other = new OtherType();
Console.WriteLine(Impromptu.InvokeMember(other, "Method", 2)); //works
Console.WriteLine(Impromptu.InvokeMember(type, "Method", 2)); //crash
}
}
public class PublicType : OtherType
{}
public class OtherType
{
private bool Method(object a)
{
return a != null;
}
}
我理解为什么存在这样的问题,我可以看到一些可能的解决方案,比如查找类,定义方法的位置,并尝试将我的对象转换为该类,但这是相当的麻烦。
有没有简单的解决方案,最好是严格依据Impromptu?
答案 0 :(得分:2)
因此,它与DLR一起工作的方式是为调用提供上下文Type
,以便它可以确定可访问的方法。即便即时使用您正在调用的对象的类型,因此它通常适用于大多数私有方法,但显然不适用于基类。
在你的情况下,你需要为即兴创建自己的上下文,这在文档UsagePrivate中提到,它适用于后期绑定类型以及接口。从文档中也不清楚,但情况是,您可以为上下文传入typeof()对象。所以在你的例子中你可以这样做:
var context = InvokeContext.CreateContext;
Console.WriteLine(Impromptu.InvokeMember(context(type, typeof(OtherType)), "Method", 2));
如果你必须为通用情况执行此操作,它并不漂亮,但是你总是可以捕获异常并递归地尝试基类型,因为它的第一次工作的一般情况下应该不会减慢,类层次结构通常不是很深,因为你只是交互式地做了一次而不是数千次,它应该没问题。
var context = InvokeContext.CreateContext;
var type = target.GetType()
while(true){
try{
Console.WriteLine(Impromptu.InvokeMember(context(target, type), "Method", 2));
break;
}catch(RuntimeBinderException ex){
type = type.BaseType;
if(type ==null)
throw ex;
}
}