溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

詳解C#反射

發(fā)布時(shí)間:2020-07-17 14:16:31 來(lái)源:億速云 閱讀:168 作者:小豬 欄目:編程語(yǔ)言

小編這次要給大家分享的是詳解C#反射,文章內(nèi)容豐富,感興趣的小伙伴可以來(lái)了解一下,希望大家閱讀完這篇文章之后能夠有所收獲。

通常,反射用于動(dòng)態(tài)獲取對(duì)象的類(lèi)型、屬性和方法等信息。今天帶你玩轉(zhuǎn)反射,來(lái)匯總一下反射的各種常見(jiàn)操作,撿漏看看有沒(méi)有你不知道的。

獲取類(lèi)型的成員

Type 類(lèi)的 GetMembers 方法用來(lái)獲取該類(lèi)型的所有成員,包括方法和屬性,可通過(guò) BindingFlags 標(biāo)志來(lái)篩選這些成員。

using System;
using System.Reflection;
using System.Linq;

public class Program
{
  public static voidMain()
  {
    var members = typeof(object).GetMembers(BindingFlags.Public |
      BindingFlags.Static | BindingFlags.Instance);
    foreach (var member in members)
    {
      Console.WriteLine($"{member.Name} is a {member.MemberType}");
    }
  }
}

輸出:

GetType is a Method
GetHashCode is a Method
ToString is a Method
Equals is a Method
ReferenceEquals is a Method
.ctor is a Constructor

GetMembers 方法也可以不傳 BindingFlags,默認(rèn)返回的是所有公開(kāi)的成員。

獲取并調(diào)用對(duì)象的方法

Type 類(lèi)型的 GetMethod 方法用來(lái)獲取該類(lèi)型的 MethodInfo,然后可通過(guò) MethodInfo 動(dòng)態(tài)調(diào)用該方法。

對(duì)于非靜態(tài)方法,需要傳遞對(duì)應(yīng)的實(shí)例作為參數(shù),示例:

class Program
{
  public static void Main()
  {
    var str = "hello";
    var method = str.GetType()
      .GetMethod("Substring", new[] {typeof(int), typeof(int)});
    var result = method.Invoke(str, new object[] {0, 4}); // 相當(dāng)于 str.Substring(0, 4)
    Console.WriteLine(result); // 輸出:hell
  }
}

對(duì)于靜態(tài)方法,則對(duì)象參數(shù)傳空,示例:

var method = typeof(Math).GetMethod("Exp");
// 相當(dāng)于 Math.Exp(2)
var result = method.Invoke(null, new object[] {2});
Console.WriteLine(result); // 輸出(e^2):7.38905609893065

如果是泛型方法,則還需要通過(guò)泛型參數(shù)來(lái)創(chuàng)建泛型方法,示例:

class Program
{
  public static void Main()
  {
    // 反射調(diào)用泛型方法
    MethodInfo method1 = typeof(Sample).GetMethod("GenericMethod");
    MethodInfo generic1 = method1.MakeGenericMethod(typeof(string));
    generic1.Invoke(sample, null);

    // 反射調(diào)用靜態(tài)泛型方法
    MethodInfo method2 = typeof(Sample).GetMethod("StaticMethod");
    MethodInfo generic2 = method2.MakeGenericMethod(typeof(string));
    generic2.Invoke(null, null);
  }
}

public class Sample
{
  public void GenericMethod<T>()
  {
    //...
  }
  public static void StaticMethod<T>()
  {
    //...
  }
}

創(chuàng)建一個(gè)類(lèi)型的實(shí)例

使用反射動(dòng)態(tài)創(chuàng)建一個(gè)類(lèi)型的實(shí)例有多種種方式。最簡(jiǎn)單的一種是用 new() 條件聲明。

使用 new 條件聲明

如果在一個(gè)方法內(nèi)需要?jiǎng)討B(tài)創(chuàng)建一個(gè)實(shí)例,可以直接使用 new 條件聲明,例如:

T GetInstance<T>() where T : new()
{
  T instance = newT();
  return instance;
}

但這種方式適用場(chǎng)景有限,比如不適用于構(gòu)造函數(shù)帶參數(shù)的類(lèi)型。

使用 Activator 類(lèi)

使用 Activator 類(lèi)動(dòng)態(tài)創(chuàng)建一個(gè)類(lèi)的實(shí)例是最常見(jiàn)的做法,示例:

Type type = typeof(BigInteger);
object result = Activator.CreateInstance(type);
Console.WriteLine(result); // 輸出:0
result = Activator.CreateInstance(type, 123);
Console.WriteLine(result); // 輸出:123

動(dòng)態(tài)創(chuàng)建泛類(lèi)型實(shí)例,需要先創(chuàng)建開(kāi)放泛型(如List<>),再根據(jù)泛型參數(shù)轉(zhuǎn)換為具象泛型(如List<string>),示例:

// 先創(chuàng)建開(kāi)放泛型
Type openType = typeof(List<>);
// 再創(chuàng)建具象泛型
Type[] tArgs = { typeof(string) };
Type target = openType.MakeGenericType(tArgs);
// 最后創(chuàng)建泛型實(shí)例
List<string> result = (List<string>)Activator.CreateInstance(target);

如果你不知道什么是開(kāi)放泛型和具象泛型,請(qǐng)看本文最后一節(jié)。

使用構(gòu)造器反射

也可以通過(guò)反射構(gòu)造器的方式動(dòng)態(tài)創(chuàng)建類(lèi)的實(shí)例,比上面使用 Activator 類(lèi)要稍稍麻煩些,但性能要好些。示例:

ConstructorInfo c = typeof(T).GetConstructor(new[] { typeof(string) });
if (c == null)
  throw new InvalidOperationException("...");
T instance = (T)c.Invoke(new object[] { "test" });

使用 FormatterServices 類(lèi)

如果你想創(chuàng)建某個(gè)類(lèi)的實(shí)例的時(shí)候不執(zhí)行構(gòu)造函數(shù)和屬性初始化,可以使用 FormatterServices 的 GetUninitializedObject 方法。示例:

class Program
{
  static void Main()
  {
    MyClass instance = (MyClass)FormatterServices.GetUninitializedObject(typeof(MyClass));
    Console.WriteLine(instance.MyProperty1); // 輸出:0
    Console.WriteLine(instance.MyProperty2); // 輸出:0
  }
}

public class MyClass
{
  public MyClass(int val)
  {
    MyProperty1 = val < 1 &#63; 1 : val;
  }

  public int MyProperty1 { get; }

  public int MyProperty2 { get; set; } = 2;
}

獲取屬性或方法的強(qiáng)類(lèi)型委托

通過(guò)反射獲取到對(duì)象的屬性和方法后,如果你想通過(guò)強(qiáng)類(lèi)型的方法來(lái)訪問(wèn)或調(diào)用,可以在中間加一層委托。這樣的好處是有利于封裝,調(diào)用者可以明確的知道調(diào)用時(shí)需要傳什么參數(shù)。 比如下面這個(gè)方法,把 Math.Max 方法提取為一個(gè)強(qiáng)類(lèi)型委托:

var tArgs = new Type[] { typeof(int), typeof(int) };
var maxMethod = typeof(Math).GetMethod("Max", tArgs);
var strongTypeDelegate = (Func<int, int, int>)Delegate
  .CreateDelegate(typeof(Func<int, int, int>), null, maxMethod);
Console.WriteLine("3 和 5 之間最大的是:{0}", strongTypeDelegate(3, 5)); // 輸出:5

這個(gè)技巧也適用于屬性,可以獲取強(qiáng)類(lèi)型的 Getter 和 Setter。示例:

var theProperty = typeof(MyClass).GetProperty("MyIntProperty");

// 強(qiáng)類(lèi)型 Getter
var theGetter = theProperty.GetGetMethod();
var strongTypeGetter = (Func<MyClass, int>)Delegate
  .CreateDelegate(typeof(Func<MyClass, int>), theGetter);
var intVal = strongTypeGetter(target); // 相關(guān)于:target.MyIntProperty

// 強(qiáng)類(lèi)型 Setter
var theSetter = theProperty.GetSetMethod();
var strongTypeSetter = (Action<MyClass, int>)Delegate
  .CreateDelegate(typeof(Action<MyClass, int>), theSetter);
strongTypeSetter(target, 5); // 相當(dāng)于:target.MyIntProperty = 5

反射獲取自定義特性

以下是四個(gè)常見(jiàn)的場(chǎng)景示例。

示例一,找出一個(gè)類(lèi)中標(biāo)注了某個(gè)自定義特性(比如 MyAtrribute)的屬性。

var props = type
  .GetProperties(BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance)
  .Where(prop =>Attribute.IsDefined(prop, typeof(MyAttribute)));

示例二,找出某個(gè)屬性的所有自定義特性。

var attributes = typeof(t).GetProperty("Name").GetCustomAttributes(false);

示例三,找出程序集所有標(biāo)注了某個(gè)自定義特性的類(lèi)。

static IEnumerable<Type> GetTypesWithAttribute(Assembly assembly)
{
  foreach(Type type inassembly.GetTypes())
  {
    if (type.GetCustomAttributes(typeof(MyAttribute), true).Length > 0)
    {
      yield return type;
    }
  }
}

示例四,在運(yùn)行時(shí)讀取自定義特性的值

public static class AttributeExtensions
{
  public static TValue GetAttribute<TAttribute, TValue>(
    this Type type,
    string MemberName,
    Func<TAttribute, TValue> valueSelector,
    bool inherit = false)
    where TAttribute : Attribute
  {
    var att = type.GetMember(MemberName).FirstOrDefault()
      .GetCustomAttributes(typeof(TAttribute), inherit)
      .FirstOrDefault() as TAttribute;
    if (att != null)
    {
      return valueSelector(att);
    }
    return default;
  }
}

// 使用:

class Program
{
  static void Main()
  {
    // 讀取 MyClass 類(lèi)的 MyMethod 方法的 Description 特性的值
    var description = typeof(MyClass)
      .GetAttribute("MyMethod", (DescriptionAttribute d) => d.Description);
    Console.WriteLine(description); // 輸出:Hello
  }
}
public class MyClass
{
  [Description("Hello")]
  public void MyMethod() { }
}

動(dòng)態(tài)實(shí)例化接口的所有實(shí)現(xiàn)類(lèi)(插件激活)

通過(guò)反射來(lái)動(dòng)態(tài)實(shí)例化某個(gè)接口的所有實(shí)現(xiàn)類(lèi),常用于實(shí)現(xiàn)系統(tǒng)的插件式開(kāi)發(fā)。比如在程序啟動(dòng)的時(shí)候去讀取指定文件夾(如 Plugins)中的 dll 文件,通過(guò)反射獲取 dll 中所有實(shí)現(xiàn)了某個(gè)接口的類(lèi),并在適當(dāng)?shù)臅r(shí)候?qū)⑵鋵?shí)例化。大致實(shí)現(xiàn)如下:

interface IPlugin
{
  string Description { get; }
  void DoWork();
}

某個(gè)在獨(dú)立 dll 中的類(lèi):

class HelloPlugin : IPlugin
{
  public string Description => "A plugin that says Hello";
  public void DoWork()
  {
    Console.WriteLine("Hello");
  }
}

在你的系統(tǒng)啟動(dòng)的時(shí)候動(dòng)態(tài)加載該 dll,讀取實(shí)現(xiàn)了 IPlugin 接口的所有類(lèi)的信息,并將其實(shí)例化。

public IEnumerable<IPlugin> InstantiatePlugins(string directory)
{
  var assemblyNames = Directory.GetFiles(directory, "*.addin.dll")
    .Select(name => new FileInfo(name).FullName).ToArray();

  foreach (var fileName assemblyNames)
    AppDomain.CurrentDomain.Load(File.ReadAllBytes(fileName));

  var assemblies = assemblyNames.Select(System.Reflection.Assembly.LoadFile);
  var typesInAssembly = assemblies.SelectMany(asm =>asm.GetTypes());
  var pluginTypes = typesInAssembly.Where(type => typeof (IPlugin).IsAssignableFrom(type));

  return pluginTypes.Select(Activator.CreateInstance).Cast<IPlugin>();
}

檢查泛型實(shí)例的泛型參數(shù)

前文提到了構(gòu)造泛型和具象泛型,這里解釋一下。大多時(shí)候我們所說(shuō)的泛型都是指構(gòu)造泛型,有時(shí)候也被稱為具象泛型。比如 List<int> 就是一個(gè)構(gòu)造泛型,因?yàn)樗梢酝ㄟ^(guò) new 來(lái)實(shí)例化。相應(yīng)的,List<> 泛型是非構(gòu)造泛型,有時(shí)候也被稱為開(kāi)放泛型,它不能被實(shí)例化。開(kāi)放泛型通過(guò)反射可以轉(zhuǎn)換為任意的具象泛型,這一點(diǎn)前文有示例。

假如現(xiàn)在有一個(gè)泛型實(shí)例,出于某種需求,我們想知道構(gòu)建這個(gè)泛型實(shí)例需要用什么泛型參數(shù)。比如某人創(chuàng)建了一個(gè) List<T> 泛型的實(shí)例,并把它作為參數(shù)傳給了我們的一個(gè)方法:

var myList = newList<int>();
ShowGenericArguments(myList);

我們的方法簽名是這樣的:

public void ShowGenericArguments(object o)

這時(shí),作為此方法的編寫(xiě)者,我們并不知道這個(gè) o 對(duì)象具體是用什么類(lèi)型的泛型參數(shù)構(gòu)建的。通過(guò)反射,我們可以得到泛型實(shí)例的很多信息,其中最簡(jiǎn)單的就是判斷一個(gè)類(lèi)型是不是泛型:

public void ShowGenericArguments(object o)
{
  if (o == null) return;
  Type t =o.GetType();
  if (!t.IsGenericType) return;
  ...
}

由于 List<> 本身也是泛型,所以上面的判斷不嚴(yán)謹(jǐn),我們需要知道的是對(duì)象是不是一個(gè)構(gòu)造泛型(List<int>)。而 Type 類(lèi)還提供了一些有用的屬性:

typeof(List<>).IsGenericType // true
typeof(List<>).IsGenericTypeDefinition // true
typeof(List<>).IsConstructedGenericType// false

typeof(List<int>).IsGenericType // true
typeof(List<int>).IsGenericTypeDefinition // false
typeof(List<int>).IsConstructedGenericType// true

IsConstructedGenericType IsGenericTypeDefinition 分別用來(lái)判斷某個(gè)泛型是不是構(gòu)造泛型和非構(gòu)造泛型。

再結(jié)合 Type 的 GetGenericArguments() 方法,就可以很容易地知道某個(gè)泛型實(shí)例是用什么泛型參數(shù)構(gòu)建的了,例如:

static void ShowGenericArguments(object o)
{
  if (o == null) return;
  Type t = o.GetType();
  if (!t.IsConstructedGenericType) return;
  foreach (Type genericTypeArgument in t.GetGenericArguments())
    Console.WriteLine(genericTypeArgument.Name);
}

看完這篇關(guān)于詳解C#反射的文章,如果覺(jué)得文章內(nèi)容寫(xiě)得不錯(cuò)的話,可以把它分享出去給更多人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI