C#學習筆記12

来源:http://www.cnblogs.com/zwt-blog/archive/2017/01/23/6344758.html
-Advertisement-
Play Games

1.在使用反射時,反射可以繞過安全訪問級別(private、protected)修飾的類或屬性,來獲取需要的信息。 2.泛型的反射:可以使用Type.ContainsGenericParameters這個屬性來判斷一個類或方法是否包含尚未設置的泛型實參,Type.IsGenericType屬性表示是 ...


1.在使用反射時,反射可以繞過安全訪問級別(private、protected)修飾的類或屬性,來獲取需要的信息。

2.泛型的反射:可以使用Type.ContainsGenericParameters這個屬性來判斷一個類或方法是否包含尚未設置的泛型實參,Type.IsGenericType屬性表示是否為泛型類型。

3.特性(Attribute):可以使用特性修飾類、介面。結構、枚舉、委托、事件、屬性、欄位、方法、構造器、索引器、參數、類型參數、返回值、程式集、模塊,使用特性的語法有2種,可為多個“[特性類型]”或“[特性類型,特性類型]”。對於其中列出多數構造來說,都可以使用上面的語法標記,但是這個語法不適合“返回值、程式集、模塊”。

(1)程式集,[assembly:特性名]。

(2)模塊,[module:特性名]。

(3)返回值,[return:特性名]。

4.大多數特性只針對特定的構造進行修飾,為了避免特性不恰當的使用,可以使用[AttributeUsageAttribute(AttributeTargets.xxx)]特性類進行標記特性限制。

5.具名參數:如[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]語法有別於構造器初始化語法,因為AttributeUsageAttribute類並不包含兩個參數的構造器,雖然C#4.0支持命名參數,但也是指定方法本身所需的參數。具名參數是用於在特性構造器調用中設置特定的公共屬性和欄位,即使構造器不包含對應的參數,具名參數雖然是可選的,但它允許對特性的額外實例數據進行設置,同時無需提供一個對應的構造器參數。

6.序列化(Serializable):本質上System.Runtime,Serialization.SerializationInfo對象是由“名稱/值”對構成的集合。在實現自定義的序列化時(需要實現ISerializable介面),內中會使用到SerializationInfo對象。

7.動態編程(Dynamic):反射的關鍵功能之一就是動態查找和調用特定類型的一個成員,這需要在執行時識別成員名或其他特征。在C#4.0中新增的動態編程-Dynamic,提供一個更簡單的辦法來通過反射調用成員,但這個技術的限制在於,編譯時需要知道成員名和簽名。若執行時發現事實上沒有這個成員,調用就會引發一個RuntimeBinderException異常。可查看UseDynamic.Test()代碼。

8.Dynamic:究其根本,Dynamic是一個Object,存在任何對象都能隱式轉換成Dynamic,Dynamic可以顯式轉換成其他對象,所以Dynamic在行為上就像Object,類似於Object,它甚至為它的預設值返回null(default(dynamic))。dynamic特殊動態行為只在調用時才會出現,這個行為是它與Object區分開來的關鍵。任何dynamic的成員調用都會返回為dynamic類型,但若對Dynamic執行GetType(),會返回是編譯好的類型(即最後賦值給動態變數的類型)。

9.實現自定義動態對象:定義自定義動態類型的關鍵是實現System.Dynamic.IDynamicMetaObjectProvider介面,但是不必從頭實現,相反首先的方案是從System.Dynamic.DynamicObject類繼承,並重寫相應的方法。DynamicObject類已經實現了IDynamicMetaObjectProvider介面,提供了預設處理。可查看代碼DynamicXml類,UseDynamic類中2個方法NormalMethod()與DynamicMethod()兩者的比較。NormalMethod()方法使用一般的xml數據讀取,DynamicMethod()方法使用了自定義動態對象解析xml內容。

[Serializable]
[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public class MyDescriptionAttribute : Attribute
{

}

public class UseDynamic
{
    public static void Test()
    {
        dynamic data = "Hello!  My name is Inigo Montoya";
        Console.WriteLine(data);
        data = (double)data.Length;
        data = data * 3.5 + 28.6;
        if (data == 2.4 + 112 + 26.2)
        {
            Console.WriteLine("data for length : {0}", data);
        }
        else
        {
            data.NonExistentMethodCallStillCompiles();
        }
    }

    public static void NormalMethod()
    {
        XElement person = XElement.Parse("<Person><Name>主神</Name><Age>26</Age></Person>");
        Console.WriteLine("{0},{1}", person.Descendants("Name").FirstOrDefault().Value, person.Descendants("Age").FirstOrDefault().Value);
    }

    public static void DynamicMethod()
    {
        dynamic person = DynamicXmL.Parse("<Person><Name>主神</Name><Age>26</Age></Person>");
        Console.WriteLine("{0},{1}", person.Name, person.Age);
    }
}

/// <summary>
/// 實現自定義的動態對象
/// </summary>
public class DynamicXmL : DynamicObject
{
    private XElement element;

    public DynamicXmL(XElement xElement)
    {
        element = xElement;
    }

    public static dynamic Parse(string text)
    {
        return new DynamicXmL(XElement.Parse(text));
    }

    public override bool TryGetMember(GetMemberBinder binder, out object result)
    {
        bool success = false;
        result = null;
        XElement firstDescendant = element.Descendants(binder.Name).FirstOrDefault();
        if (firstDescendant != null)
        {
            if (firstDescendant.Descendants().Count() > 0)
            {
                result = new DynamicXmL(firstDescendant);
            }
            else
            {
                result = firstDescendant.Value;
            }
            success = true;
        }
        return success;
    }

    public override bool TrySetMember(SetMemberBinder binder, object value)
    {
        bool success = false;
        XElement firstDescendant = element.Descendants(binder.Name).FirstOrDefault();
        if (firstDescendant != null)
        {
            if (value.GetType() == typeof(XElement))
            {
                firstDescendant.ReplaceWith(value);
            }
            else
            {
                firstDescendant.Value = value.ToString();
            }
            success = true;
        }
        return success;
    }
}
View Code

-----------------以上內容根據《C#本質論 第三版》進行整理


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

-Advertisement-
Play Games
更多相關文章
  • 要求:取指定目錄下麵的所有圖片,以表格的型式展示並顯示該圖片的相對路徑。 服務端代碼: 前端代碼: 效果圖如下: ...
  • 3. 添加一個視圖 · 原文地址:http://www.asp.net/mvc/tutorials/mvc-4/getting-started-with-aspnet-mvc4/adding-a-view · 譯文地址:http://www.cnblogs.com/powertoolsteam/ar ...
  • 2. 添加一個控制器 · 原文地址:http://www.asp.net/mvc/tutorials/mvc-4/getting-started-with-aspnet-mvc4/adding-a-controller · 譯文地址:http://www.cnblogs.com/powertools ...
  • 1.平臺互操作性和不安全的代碼:C#功能強大,但有些時候,它的表現仍然有些“力不從心”,所以我們只能摒棄它所提供的所有安全性,轉而退回到記憶體地址和指針的世界。 C#通過3種方式對此提供支持。 (1)第一種方式是通過平臺調用(Platform Invoke,P/Invoke)來調用非托管代碼DLL所公 ...
  • Socket裡面的協議解析是Socket通訊程式設計中最複雜的地方,如果你的應用層協議設計或實現不佳,Socket通訊中常見的粘包,分包就難以避免。SuperSocket內置了命令行格式的協議CommandLineProtocol,如果你使用了其它格式的協議,就必須自行實現自定義協議CustomPr ...
  • 1.在多個線程的同步數據中,避免使用this、typeof(type)、string進行同步鎖,使用這3個容易造成死鎖。 2.使用Interlocked類:我們一般使用的互斥鎖定模式(同步數據)為Lock關鍵字(即Monitor類),這個同步屬於代價非常高的一種操作。除了使用Monitor之外,還有 ...
  • 1.Task概述:Task是對操作系統線程的抽象,目的是使線程池能高效地管理線程的分配和回收,Task使用的底層線程屬於一種共用資源,任務需要互相協作,並及時歸還線程,以便用相同的共用資源(線程)滿足其他請求。 2.Task.AsyncState:獲取在創建 Task 時提供的狀態對象,如果未提供, ...
  • 1.List.BinarySearch():BinarySearch()採用的是二分搜索演算法,要求元素已經排好序,其特點是假如元素沒有找到,會返回一個負整數,該值的按位取反(~)結果是“大於被查找元素的下一個元素”的索引,如果沒有更大的值,則是元素的總數。這樣一來就可以在列表中的特定位置方便地插入新 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...