繁体   English   中英

如何从外部程序集访问内部类?

[英]How can I access an internal class from an external assembly?

有一个我无法修改(供应商提供)的程序集,它有一个返回对象类型但实际上是内部类型的方法。

如何从我的程序集中访问对象的字段和/或方法?

请记住,我无法修改供应商提供的程序集。

本质上,这就是我所拥有的:

来自供应商:

internal class InternalClass
  public string test;
end class

public class Vendor
  private InternalClass _internal;
  public object Tag {get{return _internal;}}
end class

从我使用供应商程序集的程序集。

public class MyClass
{
  public void AccessTest()
  {
    Vendor vendor = new Vendor();
    object value = vendor.Tag;
    // Here I want to access InternalClass.test
  }
}

我只看到一个案例,你允许你的内部成员接触另一个组件,这是为了测试目的。

说有一种方法允许“朋友”程序集访问内部:

在项目的AssemblyInfo.cs文件中,为每个程序集添加一行。

[assembly: InternalsVisibleTo("name of assembly here")]

此信息可在此处获得。

希望这可以帮助。

如果没有访问类型(并且没有“InternalsVisibleTo”等),则必须使用反射。 但是,一个更好的问题是:你应该可以访问这些数据? 它不是公共类型合同的一部分......它听起来像我的意图被视为一个不透明的对象(为了他们的目的,而不是你的目的)。

您已将其描述为公共实例字段; 通过反思得到这个:

object obj = ...
string value = (string)obj.GetType().GetField("test").GetValue(obj);

如果它实际上是属性(不是字段):

string value = (string)obj.GetType().GetProperty("test").GetValue(obj,null);

如果它是非公共的,则需要使用GetField / GetPropertyBindingFlags重载。

重要的是 :小心这样的反思; 实现可能会在下一个版本中发生变化(破坏您的代码),或者它可能会被混淆(破坏您的代码),或者您可能没有足够的“信任”(破坏您的代码)。 你是否发现了这种模式?

我想争论一点 - 你无法扩充原始程序集 - 使用Mono.Cecil你可以将[InternalsVisibleTo(...)]注入3pty程序集。 请注意,可能存在法律含义 - 您正在搞乱3pty汇编和技术含义 - 如果程序集具有强名称,您需要将其删除或使用不同的密钥重新签名。

 Install-Package Mono.Cecil

代码如下:

static readonly string[] s_toInject = {
  // alternatively "MyAssembly, PublicKey=0024000004800000... etc."
  "MyAssembly"
};

static void Main(string[] args) {
  const string THIRD_PARTY_ASSEMBLY_PATH = @"c:\folder\ThirdPartyAssembly.dll";

   var parameters = new ReaderParameters();
   var asm = ModuleDefinition.ReadModule(INPUT_PATH, parameters);
   foreach (var toInject in s_toInject) {
     var ca = new CustomAttribute(
       asm.Import(typeof(InternalsVisibleToAttribute).GetConstructor(new[] {
                      typeof(string)})));
     ca.ConstructorArguments.Add(new CustomAttributeArgument(asm.TypeSystem.String, toInject));
     asm.Assembly.CustomAttributes.Add(ca);
   }
   asm.Write(@"c:\folder-modified\ThirdPartyAssembly.dll");
   // note if the assembly is strongly-signed you need to resign it like
   // asm.Write(@"c:\folder-modified\ThirdPartyAssembly.dll", new WriterParameters {
   //   StrongNameKeyPair = new StrongNameKeyPair(File.ReadAllBytes(@"c:\MyKey.snk"))
   // });
}

反射。

using System.Reflection;

Vendor vendor = new Vendor();
object tag = vendor.Tag;

Type tagt = tag.GetType();
FieldInfo field = tagt.GetField("test");

string value = field.GetValue(tag);

明智地使用电源。 不要忘记错误检查。 :)

在 .NET 5 中,可以将 InternalsVisibleToAttribute 添加到您的 .csproj:

<ItemGroup>
    <AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute">
      <_Parameter1>Core.Tests</_Parameter1>
    </AssemblyAttribute>
</ItemGroup>

好吧,你做不到。 内部类在其程序集之外是不可见的,因此没有直接访问它的明确方法-AFAIK当然。 唯一的方法是通过反射使用运行时后期绑定,然后您可以间接地从内部类调用方法和属性。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM