C#反射實現

来源:https://www.cnblogs.com/loveleaf/archive/2018/11/08/9923970.html
-Advertisement-
Play Games

一、反射概念: 1、概念: 反射,通俗的講就是我們在只知道一個對象的內部而不瞭解內部結構的情況下,通過反射這個技術可以使我們明確這個對象的內部實現。 在.NET中,反射是重要的機制,它可以動態的分析程式集Assembly,模塊Module,類型Type等等,我們在不需要使用new關鍵的情況下,就可以 ...


一、反射概念:

1、概念:

    反射,通俗的講就是我們在只知道一個對象的內部而不瞭解內部結構的情況下,通過反射這個技術可以使我們明確這個對象的內部實現。

在.NET中,反射是重要的機制,它可以動態的分析程式集Assembly,模塊Module,類型Type等等,我們在不需要使用new關鍵的情況下,就可以動態

創建對象,使用對象。降低代碼耦合性提高了程式的靈活性。那麼,反射是怎麼實現的呢?它的內部實現依賴於元數據。元數據,簡單來說,在

公共語言運行時CLR中,是一種二進位信息,用來描述數據,數據的屬性環境等等的一項數據,那麼反射解析數據的內部實現通過元數據實現再

合適不過了。

2、實例:

首先先寫一個你要反射的程式集:

 

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace StudentClass
{
    public class Student
    {
        public Student()
        {

        }
        public string Name { get; set; }
        public int Age { get; set; }
        public char Gender { get; set; }
        public string IdCard { get; set; }
        public string Address { get; set; }
        private string Mobile { get; set; }
        public void Eat()
        {
            Console.WriteLine("我今天吃啦好多東西");
        }
        public void Sing()
        {
            Console.WriteLine("耶耶耶耶耶");
        }
        public int Calculate(int a, int b)
        {
            return a + b;
        }
        private string PrivateMethod()
        {
            return "我是一個私有方法";
        }
    }
}

 

先來看一下程式街、模塊、以及類等信息。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;

namespace ReflectionInvoke
{
    class Program
    {
        static void Main(string[] args)
        {
            //獲取程式集信息
            Assembly assembly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Console.WriteLine("程式集名字:"+assembly.FullName);
            Console.WriteLine("程式集位置:"+assembly.Location);
            Console.WriteLine("運行程式集需要的額CLR版本:"+assembly.ImageRuntimeVersion);
            Console.WriteLine("====================================================");
            //獲取模塊信息
            Module[] modules = assembly.GetModules();
            foreach (Module item in modules)
            {
                Console.WriteLine("模塊名稱:"+item.Name);
                Console.WriteLine("模塊版本ID"+item.ModuleVersionId);
            }
            Console.WriteLine("======================================================");
            //獲取類,通過模塊和程式集都可以
            Type[] types = assembly.GetTypes();
            foreach (Type item in types)
            {
                Console.WriteLine("類型的名稱:"+item.Name);
                Console.WriteLine("類型的完全命名:"+item.FullName);
                Console.WriteLine("類型的類別:"+item.Attributes);
                Console.WriteLine("類型的GUID:"+item.GUID);
                Console.WriteLine("=====================================================");
            }


            //獲取主要類Student的成員信息等
            Type studentType = assembly.GetType("StudentClass.Student");//完全命名
            MemberInfo[] mi = studentType.GetMembers();
            foreach (MemberInfo item in mi)
            {
                Console.WriteLine("成員的名稱:"+item.Name);
                Console.WriteLine("成員類別:"+item.MemberType);
            }
            Console.WriteLine("=====================================");

            //獲取方法
            BindingFlags flags = BindingFlags.Public | BindingFlags.DeclaredOnly | BindingFlags.Instance;
            MethodInfo[] methodInfo = studentType.GetMethods(flags);
            foreach (MethodInfo item in methodInfo)
            {
                Console.WriteLine("public類型的,不包括基類繼承的實例方法:"+item.Name);
            }
            Console.WriteLine("========================================");
            BindingFlags flag = BindingFlags.DeclaredOnly | BindingFlags.Instance | BindingFlags.NonPublic;
            MethodInfo[] methods = studentType.GetMethods(flag);
            foreach (MethodInfo item in methods)
            {
                Console.WriteLine("非public類型的,不包括基類繼承的實例方法:"+item.Name);
            }
            Console.WriteLine("========================================");

           //獲取屬性
            BindingFlags flags2 = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.DeclaredOnly | BindingFlags.Instance;
            PropertyInfo[] pi = studentType.GetProperties(flags2);
            foreach (PropertyInfo item in pi)
            {
                Console.WriteLine("屬性名稱:"+item.Name);
            }
        }
    }
}

結果:

 

 

1、Assembly.Load()以及Assembly.LoadFile():

LoadFile這個方法的參數是程式集的絕對路徑,通過點擊程式集shift+滑鼠右鍵複製路徑即可。load方法有多個重載,還可以通過流的方式獲取程式集,

在項目中,主要用來取相對路徑,因為很多項目的程式集會被生成在一個文件夾里,此時取相對路徑不容易出錯。

2、GetTypes和GetType():

很明顯第一個獲取程式集下所有的類,返回一個數組,第二個要有參數,類名為完全類名:命名空間+類名,用於獲取指定的類。

3、Type類下可以獲取這個類的所有成員,也可以獲取欄位屬性方法等,有:

ConstructorInfo獲取構造函數, FieldInfo獲取欄位, MethodInfo獲取方法,PropertyInfo獲取屬性,EventInfo獲取事件,ParameterInfo獲取參數,通過他們的

Get***獲取,加s獲取所有返回數組,不加s獲取具體的。

4、BindFlags:用於對獲取的成員的類型加以控制:

通過反編譯工具,可以看到這個enum的具體:

 BindingFlags.Public公共成員,NonPublic,非公有成員,DeclaredOnly僅僅反射類上聲明的成員不包括簡單繼承的成員。CreateInstance調用構造函數,GetField獲取欄位值對setField無效。還有很多讀者可以F12打開看一下用法以及註釋。註意必須指定:BindingFlags.Instance或BindingFlags.Static,主要為了獲取返回值,是靜態的還是實例的。

二、反射的運用:

1、創建實例:

創建實例大體分為2種,Activator.CreateInstance和Assembly.CreateInstance。這2種方法都可以創建實例,但是又有區別,下麵來通過實例具體說明。

首先分析第一種Activator.CreateInstance

這個方法有許多的重載,最常用的2種:(Type type)和(Type type,params object[] obj)第一種調用無參構造,第二種調用有參構造

在前面的實例Student中添加一個有參構造:

 public Student(string name)
        {
            this.Name = name;
        }

然後反射創建實例

Assembly assmbly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Type studentType = assmbly.GetType("StudentClass.Student");
            object obj = Activator.CreateInstance(studentType, new object[] { "milktea" });
            if (obj != null)
            {
                Console.WriteLine(obj.GetType());
            } 

這裡就創建了一個實例,現在讓我們用反編譯工具查看它的底層實現:

public static object CreateInstance(Type type, params object[] args)
        {
            return CreateInstance(type, BindingFlags.CreateInstance | BindingFlags.Public | BindingFlags.Instance, null, args, null, null);
        }

調用它的參數最多的一個重載後,發現他調用了下麵這個方法:

這裡我們就可以知道這裡創建實例和new創建實例的第三步實現相同,new創建實例,先在堆中開闢新空間,然後創建對象調用它的構造函數,

所以我們可以知道Activator.CreateInstance的底層仍然是通過被調用的類別的構造創建的,那麼如果沒有參數就說明調用的是無參構造。

然後來看第二種Assembly.CreateInstance:

Assembly assmbly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Type studentType = assmbly.GetType("StudentClass.Student");
            object o = assmbly.CreateInstance(studentType.FullName,true);
            Console.WriteLine(o.GetType());

運行程式,卻發現此時拋出了MissingMethodException異常:

可是明明有一個構造函數,為什麼還會說沒有找到構造函數呢?

通過反編譯工具,來看看什麼原因:

我們發現Assembly這個類下的CreateInstance方法,居然返回的是Activator下的CreateInstance方法,那麼就只有一種可能,他調用的

是反射類下的無參構造,而無參構造被我們新加的有參構造給替代了,因此也就找不到無參構造,為了證明結論的正確,我們把無參構造

加上,然後重新實驗:

public Student()
        {

        }

果然和我們預想的一樣,如果沒有無參構造,那麼使用Assembly類下的方法就會拋出異常。綜合2種情況,既然Assembly下的CreateInstance

也是調用的Activator的方法,並且Assembly限制更大,那我們在創建實例的時候應當還是選Activator下的方法更不容易出錯,不是嗎。

 2、調用方法,屬性賦值等

創建了實例以後,就到了實際用途,怎麼調用它的方法,怎麼給它的欄位賦值,怎麼添加一個委托事件等,現在來看。

A、第一種方法:使用Type類的InvokeMember()方法,實例如下:

Assembly assmbly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Type studentType = assmbly.GetType("StudentClass.Student");
            object o = assmbly.CreateInstance(studentType.FullName, true);
            Type instanceType = o.GetType();

            //給屬性賦值並檢查
            instanceType.InvokeMember("Name",BindingFlags.SetProperty,null,o,new object[]{"milktea"});
            string propertyValue = instanceType.InvokeMember("Name",BindingFlags.GetProperty,null,o,null).ToString();
            Console.WriteLine(propertyValue);

            //調用方法無返回值
            instanceType.InvokeMember("Eat",BindingFlags.InvokeMethod,null,o,null);
            //調用方法有返回值
            int sum = Convert.ToInt32(instanceType.InvokeMember("Calculate",BindingFlags.InvokeMethod,null,o,new object[]{2,3}));
            Console.WriteLine(sum);

幾個重要的參數:第一個方法的名稱,Enum的值,欄位SetField,方法InvokeMethod,然後選擇要使用的對象,即剛纔反射創建的實例,最後一個要

賦的值或者方法參數等必須為一個object數組。

這個方法詳情請看MSDN官方文檔:

官方文檔

B、 第二種方法:使用FiledInfo,MethodInfo...等的Invoke方法,實例如下:

Assembly assmbly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Type studentType = assmbly.GetType("StudentClass.Student");
            object o = assmbly.CreateInstance(studentType.FullName, true);
            Type instanceType = o.GetType();
//給屬性賦值並檢查
            PropertyInfo ps = instanceType.GetProperty("Age",typeof(Int32));
            ps.SetValue(o,5,null);
            PropertyInfo pi2 = instanceType.GetProperty("Age");
            Console.WriteLine(pi2.GetValue(o,null));
            //調用方法
            MethodInfo mi = instanceType.GetMethod("Calculate", BindingFlags.Instance|BindingFlags.Public);
            object obj = mi.Invoke(o, new object[] { 1, 2 });
            int result = Convert.ToInt32(mi.Invoke(o,new object[]{1,2}));
            Console.WriteLine(result);

方法的過程即先通過方法名取的方法,註意參數中的BindingFlags的2個參數都不可以丟,否則會報空引用異常,然後Invoke方法中

第一個參數為反射創建的對象,第二個參數為賦的值,或參數等。

C、第三種方法:對於反射的優化,通過使用委托:這裡我們將使用Stopwatch對比和上次同樣結果的時間:

Assembly assmbly = Assembly.LoadFile(@"E:\測試\StudentClass\StudentClass\bin\Debug\StudentClass.dll");
            Type studentType = assmbly.GetType("StudentClass.Student");
            object o = assmbly.CreateInstance(studentType.FullName, true);
            Type instanceType = o.GetType();
//給屬性賦值並檢查
            Stopwatch sw = new Stopwatch();
            sw.Start();
            PropertyInfo ps = instanceType.GetProperty("Age",typeof(int));
            ps.SetValue(o,5,null);
            PropertyInfo pi2 = instanceType.GetProperty("Age");
            Console.WriteLine(pi2.GetValue(o,null));
            Console.WriteLine("屬性沒啟用優化:"+sw.Elapsed);
            //調用方法
            sw.Reset();
            sw.Restart();
            MethodInfo mi = instanceType.GetMethod("Calculate", BindingFlags.Instance|BindingFlags.Public);
            object obj = mi.Invoke(o, new object[] { 1, 2 });
            int result = Convert.ToInt32(mi.Invoke(o,new object[]{1,2}));
            Console.WriteLine(result);
            Console.WriteLine("方法沒啟用優化:" + sw.Elapsed);
            //給屬性賦值並檢查
            sw.Reset();
            sw.Restart();
            PropertyInfo pi3 = instanceType.GetProperty("Age", typeof(int));
            var piDele = (Action<int>)Delegate.CreateDelegate(typeof(Action<int>),o,pi3.GetSetMethod());
            piDele(5);
            var result1 = (Func<int>)Delegate.CreateDelegate(typeof(Func<int>), o, pi3.GetGetMethod());
            Console.WriteLine(result1());
            Console.WriteLine("屬性啟用優化:"+sw.Elapsed);
            //調用方法
            sw.Reset();
            sw.Restart();
            MethodInfo mi2 = instanceType.GetMethod("Calculate",BindingFlags.Instance|BindingFlags.Public);
            var miDele = (Func<int, int, int>)Delegate.CreateDelegate(typeof(Func<int,int,int>),o,mi2);
            int a = miDele(1,2);
            Console.WriteLine(a);
            Console.WriteLine("方法啟用優化:"+sw.Elapsed);

這裡可以很明顯的看到使用優化以後,時間縮短了斤2/3,試想一下,這裡只用了很少的代碼,如果代碼量很多的話就可以節省更多的時間。

當然也可以看出這裡的代碼量比較大而複雜,可以說不夠漂亮簡介,用空間換取效率,Delegate.CreateDelegate()方法具體請看: 詳情鏈接

 D、現在將最後一種,.NET 4.0出現了一個新的關鍵字:dynamic,和var有點類似的感覺,但實則不同。var是語法糖,在代碼編譯期就將真正的類型

已經替換了,Visual Studio可以推斷出var的類型,而dynamic不會在編譯期檢查,被編譯為object類型,而會在運行期做檢查,並且這個效率雖然沒

有優化後的反射快,但比普通的反射也要快一些。

Stopwatch watch1 = Stopwatch.StartNew();
            Type type = Assembly.LoadFile(@"E:\C#優化實例\StudentClass\StudentClass\bin\Debug\StudentClass.dll").GetType("StudentClass.Student");
            Object o1 = Activator.CreateInstance(type,new object[]{12});
            var method1 = type.GetMethod("Add",BindingFlags.Public|BindingFlags.Instance);
            int num1 = (int)method1.Invoke(o1,new object[]{1,2});
            Console.WriteLine(num1);
            Console.WriteLine("反射耗時"+watch1.ElapsedMilliseconds);


            Stopwatch watch2 = Stopwatch.StartNew();
            Type type2 = Assembly.LoadFile(@"E:\C#優化實例\StudentClass\StudentClass\bin\Debug\StudentClass.dll").GetType("StudentClass.Student");
            dynamic o2 = Activator.CreateInstance(type, new object[] { 13 });
            int num2 = o2.Add(2,3);
            Console.WriteLine(num2);
            Console.WriteLine("dynamic耗時:"+watch2.ElapsedMilliseconds);

這裡看到是比反射要快一些,而且代碼精簡了很多。綜合考慮下來,代碼精簡度以及耗費時間,建議儘量使用dynamic關鍵字來處理反射。

 

這裡反射的主要點總結完畢,還有不全的方面請評論留言相告,感激感激                2018-11-08     17:29:28

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 我是一直致力於:.NET技術棧、WEB前端、架構設計相關的開發與管理工作,但因國內大環境影響及公司技術方向發生轉變(由.NET全部轉為JAVA),需要熟練掌握JAVA WEB相關的知識,故我也得順應潮流,” 精通 “ JAVA ^ v ^,當然.NET CORE也是我同步學習與研究的方向,甚至 ...
  • 1.主要技術以及軟體 SSM、Redis、Solr、Mysql、Nginx、Intellij、Maven、Tomcat、SVN 2.MAVEN MAVEN中有三種工程類型: 3.後臺管理系統工程搭建 後臺管理系統的工程結構如下,其中parent為父工程,common,manager,pojo,map ...
  • 網上許多教程比較晦澀難懂,本教程按照筆者(新手)自己的視角記錄,希望給大家一些幫助 1、安裝anaconda 目前比較推薦的機器學習環境為anaconda。 Anaconda指的是一個開源的Python發行版本,其包含了conda、Python等180多個科學包及其依賴項。 通過anaconda中的 ...
  • servlet生命周期 被創建:預設情況下,當servlet第一次被訪問時,由伺服器創建該對象,調用init()初始化方法,一個servlet只會被創建一次。 可以配置servlet讓其他伺服器啟動時,就被創建。 <load-on-startup>3<load-on-startup> 負數:預設值- ...
  • web緩存: 1.可以自動保存常見文檔副本的HTTP設備,當web請求抵達緩存時,如果存在緩存副本,就直接從本地存儲設備返回,而不是去源伺服器獲取 2.緩存命中和未命中 3.HTTP再驗證,檢測伺服器上的內容是否發生了變化,新鮮度檢測規則。緩存對副本進行再驗證時,會向伺服器發送一個小的再驗證請求,如... ...
  • 冒泡排序是一個經典的案例 實現原理就數與數前後兩兩比較,如果前面比後面大則交換位置。最終達到從小到大的順序,這樣的排序方式就是冒泡排序。 ...
  • 在優化C#代碼或對比某些API的效率時,通常需要測試某個方法的運行時間,可以通過DateTime來統計指定方法的執行時間,也可以使用命名空間System.Diagnostics中封裝了高精度計時器QueryPerformanceCounter方法的Stopwatch類來統計指定方法的執行時間: 1. ...
  • 問題: 如何實現一個具有非同步簽名的同步方法。 從非同步介面或基類繼承代碼,但希望用同步方式實現方法。 解釋一下所謂的非同步介面和非同步基類。例如如下代碼 即,介面和基類中包含了以Task為返回結果的方法,此介面或基類,即為非同步介面或非同步基類。 同步方式實現Task方法 或者 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...