C#動態調用泛型類、泛型方法

来源:https://www.cnblogs.com/smiler/archive/2018/12/26/10177356.html
-Advertisement-
Play Games

在製作一個批量序列化工具時遇到瞭如下問題,在此記錄一下,僅供參考。 主程式載入另一個程式集,將其中的所有類取出,然後對這些類分別調用泛型類或泛型方法。控制台程式解決方案如下: Main工程:提供Worker類進行數據操作,XMLTool<T>泛型類將數據集序列化為.xml文檔,RootCollect ...


在製作一個批量序列化工具時遇到瞭如下問題,在此記錄一下,僅供參考。

      主程式載入另一個程式集,將其中的所有類取出,然後對這些類分別調用泛型類或泛型方法。控制台程式解決方案如下:

  • Main工程:提供Worker類進行數據操作,XMLTool<T>泛型類將數據集序列化為.xml文檔,RootCollection<T>類封裝數據集
    • Worker類

           提供成員方法void DoWork<T>()、List<T> GetList<T>()、靜態成員方法StaticDoWork<T>(),代碼如下:

    複製代碼
     1 public class Worker
    2 {
    3 public Worker()
    4 {
    5 }
    6
    7 public void DoWork<T>()
    8 {
    9 Type t = typeof(T);
    10 Console.WriteLine("Get Class: {0}", t.Name);
    11 PropertyInfo[] properties = t.GetProperties();
    12 foreach (PropertyInfo property in properties)
    13 {
    14 Console.WriteLine("\tproperty.Name: " + property.Name + "\tproperty.MemberType: " + property.PropertyType);
    15 }
    16 }
    17
    18 public static void StaticDoWork<T>()
    19 {
    20 Type t = typeof(T);
    21 Console.WriteLine("Get Class: {0}", t.Name);
    22 PropertyInfo[] properties = t.GetProperties();
    23 foreach (PropertyInfo property in properties)
    24 {
    25 Console.WriteLine("\tproperty.Name: " + property.Name + "\tproperty.MemberType: " + property.PropertyType);
    26 }
    27 }
    28
    29 public List<T> GetList<T>()
    30 {
    31 Console.WriteLine("Generate List for [{0}]", typeof(T).Name);
    32 return new List<T>()
    33 {
    34 Activator.CreateInstance<T>(),
    35 Activator.CreateInstance<T>()
    36 };
    37 }
    38 }
    複製代碼

     

    • XMLTool<T>類 複製代碼
       1publicclass XMLTool<T>
      2 {
      3publicstaticvoid XmlSerialize_Save(List<T> needSerializedList, string xmlDirPath, string xmlFileName)
      4 {
      5 RootCollection<T> collection = new RootCollection<T>();
      6 collection.ItemList = needSerializedList;
      7if (!Directory.Exists(xmlDirPath))
      8 Directory.CreateDirectory(xmlDirPath);
      9using (System.IO.FileStream stream = new System.IO.FileStream(xmlFileName, System.IO.FileMode.Create))
      10 {
      11 System.Xml.Serialization.XmlSerializer serializer = new System.Xml.Serialization.XmlSerializer(collection.GetType());
      12 serializer.Serialize(stream, collection);
      13 }
      14 }
      15 }
      複製代碼
    • RootCollection<T>類: 複製代碼
       1     [Serializable]
      2 public class RootCollection<T>
      3 {
      4 public RootCollection()
      5 {
      6 itemList = new List<T>();
      7 }
      8
      9 private List<T> itemList;
      10
      11 public List<T> ItemList
      12 {
      13 get { return itemList; }
      14 set { itemList = value; }
      15 }
      16 }
      複製代碼
  • MockClassLib工程:提供BaseEntityAppleCatPerson
    • BaseEntity類:抽象類,負責初始化類成員 複製代碼
       1     public abstract class BaseEntity
      2 {
      3 public BaseEntity()
      4 {
      5 InitiaWithNull();
      6 }
      7
      8 private void InitiaWithNull()
      9 {
      10 Type type = this.GetType();
      11 PropertyInfo[] properties = type.GetProperties();
      12 string[] PropNames = new string[properties.Length];
      13 Dictionary<string, PropertyInfo> PropNameToInfo = new Dictionary<string, PropertyInfo>();
      14 for (int i = 0; i < properties.Length; i++)
      15 {
      16 PropNames[i] = properties[i].Name;
      17 PropNameToInfo.Add(PropNames[i], properties[i]);
      18 }
      19
      20 foreach (string propname in PropNames)
      21 {
      22 string proptype = PropNameToInfo[propname].PropertyType.Name;
      23
      24 object value = null;
      25 if (NullValue.Keys.Contains(proptype))
      26 value = NullValue[proptype];
      27
      28 type.GetProperty(propname).SetValue(this, value, null);
      29 }
      30 }
      31
      32 private static readonly Dictionary<string, object> NullValue = new Dictionary<string, object>()
      33 {
      34 { "String", String.Empty },
      35 { "DateTime", DateTime.MinValue},
      36 { "Decimal", Decimal.MinValue}
      37 };
      38 }
      複製代碼
    • AppleCatPerson類:測試類,繼承於BaseEntity 複製代碼
       1     public class Apple : BaseEntity
      2 {
      3 public string Color { get; set; }
      4 }
      5
      6 public class Cat : BaseEntity
      7 {
      8 public string Type { get; set; }
      9 }
      10
      11 public class Person : BaseEntity
      12 {
      13 public int ID { get; set; }
      14 public string Name { get; set; }
      15 }
      複製代碼

 

      Main工程的Program的Main方法中,一般情況下,調用Worker的泛型方法來處理測試類的話,可以寫為:

      Worker worker = new Worker();

      worker.DoWork<Apple>();

      worker.DoWork<Cat>();

      worker.DoWork<Person>();

      但是,如果MockClassLib中需要處理的類型非常多時,這樣顯示調用必然是不靈活的,應當怎樣向泛型方法DoWork<T>()的尖括弧中動態傳入類型呢?

      考慮代碼:

複製代碼
            //Load assembly
Assembly mockAssembly = Assembly.LoadFrom("MockClassLibrary.dll");
Type[] typeArray = mockAssembly.GetTypes();

//Create instance of Worker
Worker worker = new Worker();
foreach(Type curType in typeArray)
{
worker.DoWork<curType>(); //Error
}
複製代碼

      可以看到,Type類型的實例是無法直接傳入泛型方法的尖括弧中的,T要求顯式指明類型名。

      下麵通過反射方式來獲取泛型方法,並創建特定類型的泛型方法。

  • 對於非靜態方法:public void DoWork<T>()

          對於非靜態方法,調用MethodInfo.Invoke(object, object[])時,第一個參數需要指明泛型方法的所有者(即這裡創建的worker對象),第二個參數為泛

          型方法的參數列表,DoWork<T>()沒有輸入參數,所以設為null

複製代碼
//Create an instance of Worker
Worker worker = new Worker();

//Get type of Worker
Type workerType = typeof(Worker);

//Get Generic Method
MethodInfo doWorkMethod = workerType.GetMethod("DoWork");

//Invoke DoWork<T> with different Type
foreach (Type curType in typeArray)
{
if (curType.IsClass && !curType.IsAbstract)//Filter BaseEntity
{
MethodInfo curMethod = doWorkMethod.MakeGenericMethod(curType);
curMethod.Invoke(worker, null);//Member method,use instance
}
}
複製代碼
  • 對於靜態方法:public static void StaticDoWork<T>()

          不同於非靜態方法,這裡直接反射的類靜態方法,所以Invoke()的第一個參數設為null

複製代碼
//Get type of Worker
Worker worker = new Worker();

//Get Generic Method
MethodInfo staticDoWorkMethod = workerType.GetMethod("StaticDoWork");

//Invoke StaticDoWork<T>
foreach (Type curType in typeArray)
{
if (curType.IsClass && !curType.IsAbstract)
{
MethodInfo curMethod = staticDoWorkMethod.MakeGenericMethod(curType);
curMethod.Invoke(null, null);//Static method
}
}
複製代碼
  • 對於有返回值的非靜態方法:public List<T> GetList()

          如同動態調用DoWork<T>()方法一樣,只是在處理返回值時,可以使用下麵的方法

1 IList tempList = (IList)curMethod.Invoke(worker, null);
2 //Or
3 IEnumerable tempList = (IEnumerable)curMethod.Invoke(worker, null);
  • 對於泛型類:XMLTool<T>

          下麵要使用泛型類XMLTool<T>的靜態方法public static void XmlSerialize_Save(List<T> list, string dirPath, string fileName)方法。

          首先應通過反射構造出指定類型的泛型類XMLTool<T>,再反射出其中的XmlSerialize_Save方法並使用。

複製代碼
 1 //Use Generic Class
2 Type xmlToolType = typeof(XMLTool<>).MakeGenericType(curType);
3
4 //Get method
5 MethodInfo saveMethod = xmlToolType.GetMethod("XmlSerialize_Save");
6
7 //Invoke
8 saveMethod.Invoke
9 (
10 null, //Static method
11 new object[] { resultList, @"c:\", @"c:\Test_" + curType.Name + ".xml" }

12 );
複製代碼

 

       Program-->Main()方法的全部代碼:

複製代碼
 1 namespace RetrieveUnknownClass
2 {
3 class Program
4 {
5 static void Main(string[] args)
6 {
7 //Load assembly
8 Assembly mockAssembly = Assembly.LoadFrom("MockClassLibrary.dll");
9 Type[] typeArray = mockAssembly.GetTypes();
10
11 //Create instance of Worker
12 Type workerType = typeof(Worker);
13 Worker worker = new Worker();
14
15 #region Member method
16
17 Console.WriteLine(">>>>>>>>>Use Generic Method:");
18 MethodInfo doWorkMethod = workerType.GetMethod("DoWork");
19
20 //Invoke DoWork<T>
21 foreach (Type curType in typeArray)
22 {
23 if (curType.IsClass && !curType.IsAbstract)
24 {
25 MethodInfo curMethod = doWorkMethod.MakeGenericMethod(curType);
26 curMethod.Invoke(worker, null);//Member method,use instance
27 }
28 }
29
30 #endregion
31
32 #region Static method
33
34 Console.WriteLine("\r\n>>>>>>>>>Use Static Generic Method:");
35 MethodInfo staticDoWorkMethod = workerType.GetMethod("StaticDoWork");
36
37 //Invoke StaticDoWork<T>
38 foreach (Type curType in typeArray)
39 {
40 if (curType.IsClass && !curType.IsAbstract)
41 {
42 MethodInfo curMethod = staticDoWorkMethod.MakeGenericMethod(curType);
43 curMethod.Invoke(null, null);//Static method
44 }
45 }
46
47 #endregion
48
49 #region Get A List & Serialize It to Xml File With Generic
50
51 Console.WriteLine("\r\n>>>>>>>>>Get List By Generic Method:");
52 MethodInfo getListMethod = workerType.GetMethod("GetList");
53
54 foreach (Type curType in typeArray)
55 {
56 if (curType.IsClass && !curType.IsAbstract)
57 {
58 MethodInfo curMethod = getListMethod.MakeGenericMethod(curType);
59 //Generate List
60 IList resultList = (IList)curMethod.Invoke(worker, null);
61 //Show List
62 ShowList(resultList);
63 //Use Generic Class
64 Type xmlToolType = typeof(XMLTool<>).MakeGenericType(curType);
65 MethodInfo saveMethod = xmlToolType.GetMethod("XmlSerialize_Save");
66
67 saveMethod.Invoke
68 (
69 null, //Static method
70 new object[] { resultList, @"c:\", @"c:\Test_" + curType.Name + ".xml" }
71 );
72 }
73 }
74
75 Console.WriteLine("Serialization Completed...\r\n");
76 #endregion
77 }
78
79 public static void ShowList(IList list)
80 {
81 Console.WriteLine("Type of list: {0}\r\nCount of current list: {1}\r\nType of item in list: {2}\r\n",
82 list.GetType(),
83 list.Count,
84 list[0].GetType());
85 }
86 }
87 }
複製代碼
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • XML為可擴展標記語言,它主要是用來保存數據,做配置文件,數據傳輸載體等。其實就是一個尾碼名為.xml的文件。 XML命名規則 名稱可以含字母、數字以及其他的字元 名稱不能以數字或者標點符號開始 名稱不能以字元 “xml”(或者 XML、Xml)開始 名稱不能包含空格 文檔聲明 例如此聲明:<?xm ...
  • 初入博客園,請各位多關照,來而不往非禮也。 如需要源碼以及學習內容,qq:1397617269,我看到就回你們資源。 ...
  • 首先 要想Hadoop能夠很流暢的Python程式,學習mrjob可能是最直接、最簡單的方法了 你甚至都不要按安裝部署Hadoop集群。 mrjob擁有很多優秀的特性比如: 支持多步驟的MapReduce任務工作流 支持內嵌、本地、遠程亞馬遜以及Hadoop 調試方便不需要任務環境支持 本教程通過 ...
  • Hadoop 1.0 到 Hadoop 2.0 經歷了什麼,我們又能從中看出什麼呢? ...
  • (一)python金融數據爬蟲項目 爬取目標:雪球網(起始url:https://xueqiu.com/hq#exchange=CN&firstName=1&secondName=1_0&page=1) 爬取內容:雪球網深滬股市情況 使用工具:requests庫實現發送請求、獲取響應。 json格式 ...
  • @[toc] "Django官網下載" MVC框架與MTV框架 MVC,全名Model View Controller,是軟體工程中的一種軟體架構模式. 把軟體系統分為三個基本部分:==模型(Model)、視圖(View)、控制器(Controller).== 優點:耦合性低、重用行高、生命周期成本 ...
  • 前言 出於對數據分析及數據挖掘的興趣,我開始接觸Python,由於缺乏編程相關知識,所以過程滿是曲折,也缺乏足量的實踐練習。寫這一系列文章的原因首先是為了鍛煉自己,因為我相信費曼的學習方法:要在教學中才能真正掌握所學的知識;其次也是想藉此機會給未入門的朋友們一些門檻比較低的參考資料,所以本系列文章均 ...
  • laravel項目顯示:No application encryption key has been specified 解決辦法: 項目根目錄下執行 php artisan key:generate ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...