2

给定以下课程:

class SomeBuilder<T>
{
    public static object Build(int index)
    {
        ...
    }
}

class SomeHelper
{
    public object GetBuildObj(object value)
    {
        var valuetype = value.GetType();
        var methodinfo = typeof(SomeBuilder<>).MakeGenericType(valuetype).GetMethod("Build");
        var handler = SomeDynamicHelper.GetMethodInvoker(methodinfo);
        var result = hander(null, new object[]{1});
    }
}

SomeBuilder 是一个泛型类型,所以我需要调用 MakeGenericType() 来解决问题。

当我为值传递一个像“类人”这样的普通类型时,一切正常,没关系。

但是当我传递一个匿名类型时:new { id=1 },处理程序已成功创建。但是调用这个动态处理程序我得到了一个带有这些消息的 MethodAccessException:

"method "SomeDynamicHelper.(System.Object, System.Objec[])"尝试访问方法"SomeBuilder'1<<>f__AnonymousType0'1<System.Int32>>.Build(int)"失败。

任何帮助将不胜感激,谢谢。

顺便说一句,如果您对 SomeDynamicHelper 感兴趣,请参阅:

http://www.codeproject.com/Articles/14593/A-General-Fast-Method-Invoker

编辑1:

我在 main 中进行了这样的调用:

static void Main(string[] args)
{
    // pass a normal class, this will be fine
    var value = new Person { id = 1};
    new SomeHelper().GetBuildObj(value);  

    // pass a anonymous type
    var value = new { id = 1};
    new SomeHelper().GetBuildObj(value);  // oops, got a exception here!
}

编辑2:

根据评论我改变了我的代码:

class SomeHelper
{
    public object GetBuildObj(object value)
    {
        //this time i do not use the value, but create a new inner value:
        var valuenew = new { id = 1 };
        var valuetype = valuenew.GetType();
        var methodinfo = typeof(SomeBuilder<>).MakeGenericType(valuetype).GetMethod("Build");
        var handler = SomeDynamicHelper.GetMethodInvoker(methodinfo);
        var result = hander(null, new object[]{1});
    }
}

好吧,这次也不例外,但是……不幸的是出现了一个新问题……也许我应该为新问题打开一个新线程。

谢谢你们,你们的关注。

编辑3:

嗨,经过一番挖掘,我还发现了一些有用的信息。比如说, SomeDynamicHelper.GetMethodInvoker() 代码如下:

DynamicMethod dynamicMethod = new DynamicMethod(string.Empty, typeof(object), new Type[] { typeof(object), typeof(object[]) }, methodInfo.DeclaringType.Module);

这是我们在这里用来动态创建方法的核心。对于我们的上下文,我们需要在与 SomeHelper 和 SomeBuilder 相同的程序集中声明匿名类型。但是,如果我们不能做声明,我们该怎么办?

好的,您可以使用最后一个参数(skipVisibility)调用DynamicMethod(),设置为true!

希望这会帮助其他有同样问题的人:)

4

3 回答 3

2

“方法“SomeDynamicHelper.(System.Object, System.Objec[])”尝试访问方法“SomeBuilder'1<<>f__AnonymousType0'1>.Build(int)”

从这里你可以看到动态方法尝试运行一个名为 Build 的内部\私有方法,因为你得到了MethodAccessException

(匿名类型保存在新生成的类中)

添加InternalVisibleTo并不总是有帮助,因为它只有在您可以重写匿名类型程序集(手动或使用拦截)并且只有当类型和方法是内部而不是私有时才有帮助。

在动态方法中,您可以将方法绑定到类型\模块,并跳过可见性检查,但这有助于访问指定模块的私有成员,因此如果您尝试访问的类型在不同的程序集中并且类型\方法是私人的,你无能为力。

差不多吧。您将一个“秘密”属性注入到您的程序集中,然后调用IgnoreAccessChecksTo,然后除了有界模块和跳过可见性之外,它可能会起作用。

于 2016-05-08T17:28:13.227 回答
0

您可以尝试[assembly: InternalsVisibleTo("Anonymously Hosted DynamicMethods Assembly")]在定义匿名类型的同一个项目中使用。

于 2013-06-16T14:21:19.193 回答
0

我尝试使用 IKVM 转换的 Guice 注入的 C# 类也有类似的错误。修复只是将受影响的类(在这种情况下可能SomeBuilder)公开,一切正常。

于 2016-09-27T12:51:42.867 回答