C#—反射(反射類型、方法、構造函數、屬性、實現可配置可擴展、資料庫訪問類反射封裝)

来源:http://www.cnblogs.com/AnkerZhang/archive/2017/06/30/7099664.html
-Advertisement-
Play Games

反射是.NET中的重要機制,通過反射,可以在運行時獲得程式或程式集中每一個類型(包括類、結構、委托、介面和枚舉等)的成員和成員的信息。有了反射,即可對每一個類型瞭如指掌。另外我還可以直接創建對象,即使這個對象的類型在編譯時還不知道。 Anker_張(博客園)http://www.cnblogs.co ...


反射是.NET中的重要機制,通過反射,可以在運行時獲得程式或程式集中每一個類型(包括類、結構、委托、介面和枚舉等)的成員和成員的信息。有了反射,即可對每一個類型瞭如指掌。另外我還可以直接創建對象,即使這個對象的類型在編譯時還不知道。               

                              ------------------------Anker_張(博客園)http://www.cnblogs.com/AnkerZhang/

 

反射的用途:
    1)使用Assembly定義和載入程式集,載入在程式集清單中列出模塊,以及從此程式集中查找類型並創建該類型的實例。 
    2)使用Module瞭解包含模塊的程式集以及模塊中的類等,還可以獲取在模塊上定義的所有全局方法或其他特定的非全局方法。 
    3)使用ConstructorInfo瞭解構造函數的名稱、參數、訪問修飾符(如pulic private)和實現詳細信息(如abstractvirtual)等。 
    4)使用MethodInfo瞭解方法的名稱、返回類型、參數、訪問修飾符(如pulic private)和實現詳細信息(如abstractvirtual)等。
    5)使用FiedInfo瞭解欄位的名稱、訪問修飾符(如publicprivate)和實現詳細信息(如static)等,並獲取或設置欄位值。
    6)使用EventInfo瞭解事件的名稱、事件處理程式數據類型、自定義屬性、聲明類型和反射類型等,添加或移除事件處理程式。 
    7)使用PropertyInfo瞭解屬性的名稱、數據類型、聲明類型、反射類型和只讀或可寫狀態等,獲取或設置屬性值。 
    8)使用ParameterInfo瞭解參數的名稱、數據類型、是輸入參數還是輸出參數,以及參數在方法簽名中的位置等。

 

反射用到的命名空間:

   System.Reflection
    System.Type
    System.Reflection.Assembly

反射用到的主要類:
    System.Type 類--通過這個類可以訪問任何給定數據類型的信息。
    System.Reflection.Assembly類--它可以用於訪問給定程式集的信息,或者把這個程式集載入到程式中。
System.Type類:
    System.Type 類對於反射起著核心的作用。但它是一個抽象的基類,Type有與每種數據類型對應的派生類,我們使用這個派生類的對象的方法、欄位、屬性來查找有關該類型的所有信息。

 

獲取給定類型的Type引用有3種常用方式:
    使用 C# typeof 運算符。
        Type t = typeof(string);
    ●使用對象GetType()方法。
        string s = "grayworm";
        Type t = s.GetType(); 
    ●還可以調用Type類的靜態方法GetType()
        Type t = Type.GetType("System.String");
       
    上面這三類代碼都是獲取string類型的Type,在取出string類型的Type引用t後,我們就可以通過t來探測string類型的結構了。

 

string n = "grayworm";
Type t = n.GetType();
foreach (MemberInfo mi in t.GetMembers())
{
  Console.WriteLine("{0}/t{1}",mi.MemberType,mi.Name);
}

 

 Type類的屬性:
        Name 數據類型名
        FullName 數據類型的完全限定名(包括命名空間名)
        Namespace 定義數據類型的命名空間名
        IsAbstract 指示該類型是否是抽象類型
        IsArray   指示該類型是否是數組
        IsClass   指示該類型是否是類
        IsEnum   指示該類型是否是枚舉
        IsInterface    指示該類型是否是介面
        IsPublic 指示該類型是否是公有的
        IsSealed 指示該類型是否是密封類
        IsValueType 指示該類型是否是值類型
    Type類的方法:
        GetConstructor(), GetConstructors():返回ConstructorInfo類型,用於取得該類的構造函數的信息
        GetEvent(), GetEvents():返回EventInfo類型,用於取得該類的事件的信息
        GetField(), GetFields():返回FieldInfo類型,用於取得該類的欄位(成員變數)的信息
        GetInterface(), GetInterfaces():返回InterfaceInfo類型,用於取得該類實現的介面的信息
        GetMember(), GetMembers():返回MemberInfo類型,用於取得該類的所有成員的信息
        GetMethod(), GetMethods():返回MethodInfo類型,用於取得該類的方法的信息
        GetProperty(), GetProperties():返回PropertyInfo類型,用於取得該類的屬性的信息
    可以調用這些成員,其方式是調用TypeInvokeMember()方法,或者調用MethodInfo, PropertyInfo和其他類的Invoke()方法。 

 

查看類中的構造方法:

     NewClassw nc = new NewClassw();
        Type t = nc.GetType();
        ConstructorInfo[] ci = t.GetConstructors();    //獲取類的所有構造函數
        foreach (ConstructorInfo c in ci) //遍歷每一個構造函數
        {
            ParameterInfo[] ps = c.GetParameters();    //取出每個構造函數的所有參數
            foreach (ParameterInfo pi in ps)   //遍歷並列印所該構造函數的所有參數
            {
                Console.Write(pi.ParameterType.ToString()+" "+pi.Name+",");
            }
            Console.WriteLine();
        }

用構造函數動態生成對象:

        Type t = typeof(NewClassw);
        Type[] pt = new Type[2];
        pt[0] = typeof(string);
        pt[1] = typeof(string);
        //根據參數類型獲取構造函數 
        ConstructorInfo ci = t.GetConstructor(pt); 
        //構造Object數組,作為構造函數的輸入參數 
        object[] obj = new object[2]{"grayworm","hi.baidu.com/grayworm"};   
        //調用構造函數生成對象 
        object o = ci.Invoke(obj);    
        //調用生成的對象的方法測試是否對象生成成功 
        //((NewClassw)o).show();          

Activator生成對象:

        Type t = typeof(NewClassw);
        //構造函數的參數 
        object[] obj = new object[2] { "grayworm", "hi.baidu.com/grayworm" };   
        //用Activator的CreateInstance靜態方法,生成新對象 
        object o = Activator.CreateInstance(t,"grayworm","hi.baidu.com/grayworm"); 
        //((NewClassw)o).show();

查看類中的屬性:

        NewClassw nc = new NewClassw();
        Type t = nc.GetType();
        PropertyInfo[] pis = t.GetProperties();
        foreach(PropertyInfo pi in pis)
        {
            Console.WriteLine(pi.Name);
        }

查看類中的public方法:

        NewClassw nc = new NewClassw();
        Type t = nc.GetType();
        MethodInfo[] mis = t.GetMethods();
        foreach (MethodInfo mi in mis)
        {
            Console.WriteLine(mi.ReturnType+" "+mi.Name);
        }

查看類中的public欄位:

 

        NewClassw nc = new NewClassw();
        Type t = nc.GetType();
        FieldInfo[] fis = t.GetFields();
        foreach (FieldInfo fi in fis)
        {
            Console.WriteLine(fi.Name);
        } (http://hi.baidu.com/grayworm

 

用反射生成對象,並調用屬性、方法和欄位進行操作 

        NewClassw nc = new NewClassw();
        Type t = nc.GetType();
        object obj = Activator.CreateInstance(t);
        //取得ID欄位 
        FieldInfo fi = t.GetField("ID");
        //給ID欄位賦值 
        fi.SetValue(obj, "k001");
        //取得MyName屬性 
        PropertyInfo pi1 = t.GetProperty("MyName");
        //給MyName屬性賦值 
        pi1.SetValue(obj, "grayworm", null);
        PropertyInfo pi2 = t.GetProperty("MyInfo");
        pi2.SetValue(obj, "hi.baidu.com/grayworm", null);
        //取得show方法 
        MethodInfo mi = t.GetMethod("show");
        //調用show方法 
        mi.Invoke(obj, null);

System.Reflection.Assembly 
     Assembly類可以獲得程式集的信息,也可以動態的載入程式集,以及在程式集中查找類型信息,並創建該類型的實例。
    使用Assembly類可以降低程式集之間的耦合,有利於軟體結構的合理化。
    通過程式集名稱返回Assembly對象
        Assembly ass = Assembly.Load("ClassLibrary831");
    通過DLL文件名稱返回Assembly對象
        Assembly ass = Assembly.LoadFrom("ClassLibrary831.dll");
    通過Assembly獲取程式集中類 
        Type t = ass.GetType("ClassLibrary831.NewClass");   //參數必須是類的全名
    通過Assembly獲取程式集中所有的類
        Type[] t = ass.GetTypes();

//通過程式集的名稱反射
    Assembly ass = Assembly.Load("ClassLibrary831");
    Type t = ass.GetType("ClassLibrary831.NewClass");
    object o = Activator.CreateInstance(t, "grayworm", "http://hi.baidu.com/grayworm");
    MethodInfo mi = t.GetMethod("show");
    mi.Invoke(o, null);
   //通過DLL文件全名反射其中的所有類型
    Assembly assembly = Assembly.LoadFrom("xxx.dll的路徑");
    Type[] aa = a.GetTypes();
    foreach(Type t in aa)
    {
        if(t.FullName == "a.b.c")
        {
            object o = Activator.CreateInstance(t);
        }
    }

 一下擴展黑科技

//-------------------反射-------------------------------
            Assembly assembly = Assembly.Load("Ruanmou.DB.Sqlserver");
            Type type = assembly.GetType("Ruanmou.DB.Sqlserver.ReflectionTest");
            object obj = Activator.CreateInstance(type);
            foreach (MethodInfo method in type.GetMethods())
            {
                Console.WriteLine("名稱{0}", method.Name);
            }
            MethodInfo show1 = type.GetMethod("Show1");
            Console.WriteLine(show1.Invoke(obj, null));
            MethodInfo show = type.GetMethod("Show", new Type[] { });
            show.Invoke(obj, null);
            MethodInfo shows = type.GetMethod("Show", new Type[] { typeof(string) });
            shows.Invoke(obj, new object[] { "重載方法" });
            //破壞單例模式
            Type stype = typeof(Singleton);
            object sobj = Activator.CreateInstance(stype, true);
            Console.ReadLine();
            //黑科技執行私有方法
            MethodInfo Show2 = type.GetMethod("Show2", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
            Show2.Invoke(obj, null);
            //讀取私有靜態方法
            MethodInfo show4 = type.GetMethod("Show4", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Static);
            show4.Invoke(obj, null);
            //讀取私有靜態方法帶參數
            //未知------
            //給類型賦值
            MethodInfo setid = type.GetMethod("set_Id");
            Console.WriteLine(setid.Invoke(obj, new object[] { 5 }));
            MethodInfo getid = type.GetMethod("get_Id");
            Console.WriteLine(getid.Invoke(obj, null));
            foreach (PropertyInfo p in type.GetProperties())
            {
                Console.WriteLine("共有屬性" + p.Name);
            }
            PropertyInfo number = type.GetProperty("number", BindingFlags.Instance | BindingFlags.NonPublic);
            Console.WriteLine(number.PropertyType);
            number.SetValue(obj, 15);
            Console.WriteLine(number.GetValue(obj, null));
            Console.ReadLine();

 


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

-Advertisement-
Play Games
更多相關文章
  • 鑒於之前學習的是Java,學習Java的時候一直使用的是近乎神級的開發軟體IDEA,出於對IDEA的崇拜,轉到C#大陣營後,瞭解到JetBrains公司也有相應的Vs插件, 就果斷安裝了Resharper(代碼生成工具), 在此總結了Vs2015中安裝ReSharper的步驟: 1.首先必須安裝Vi ...
  • 現在有一個發佈好的WebService,地址是http://hovertree.com:2706/UploadExpenseToConstract.asmx能不能把最後面的asmx去掉呢?變成http://hovertree.com:2706/UploadExpenseToConstract但效果和 ...
  • 前言 前幾天在博客園看到有園友在分享關於微軟的一個微服務架構的示常式序,想必大家都已經知道了,那就是 "eShopOnContainers" 。 我們先不看項目的尾碼名稱 OnXXX ,因為除了 OnContainers 還有 OnAzure,OnWeb,OnKubernetes 以及 OnServ ...
  • 一、請求從路由開始 1.為什麼需要路由? (1).屏蔽物理路徑、提高安全性 (2).有利於搜索引擎優化 2.定義路由的規則 (1).基於模式匹配的路由規則 語法:{占位符1}字面量1{占位符2}字面量2...{占位符n}...字面量n 註:字面量可能是一個固定的字元,比較常見的是"/",也可以是一個 ...
  • 物性路由:將路由和控制器放在一起,這樣更簡單方便,還可以處理複雜的路由場景 傳統路由:集中、強制、基於代碼風格來定義的。 每個MVC應用程式都需要路由來定義自己的處理請求方式,路由是MVC是應用程式的入口點。我們先瞭解一下路由涉及的主要概念:路由定義是從URL模板開始的,因為它指定了與路由相匹配的模 ...
  • 百度翻譯是什麼,可以吃嗎?相信很多人都熟悉,它是我們生活中必不可少的一隻東東。 但是,百度翻譯開發平臺只有每月只能翻譯200萬個字元,多出的要按照49.00/百萬字元來算。對於我醬紫的乞丐程式員來說,其實已經是足足的了。 接下來進入正題,分為2個部分講,一是免費的百度翻譯,二是收費的百度翻譯(此處應 ...
  • 本章旨在講解外掛實現原理,未深入涉及至代碼層面。希望能與對這方面感興趣的朋友多多交流,畢竟理論是死的,套路是固定的,只有破解經驗是花大量時間和心血積累的。 對於單機游戲而言,游戲中絕大部分的參數(比如血、藍、能量亦或是金幣)都存儲在電腦的堆棧中,一些類似劇情進度的則加密後寫入本地的自定義配置文件中 ...
  • 之前公司下達了開發微信只一塊的任務,然後我就去網上取了一番經,可是感覺對於我這種微信開發的新手來說,所需要的東西太多,太凌亂了,然後整個人就懵逼了。 費了很長時間才將微信介面調用完成。所以吶,我整理了一下,調用微信介面前需要準備的內容。 1.微信公眾平臺的appid 2.微信公眾平臺的secret ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...