[.NET] 《Effective C#》快速筆記 - C# 中的動態編程

来源:http://www.cnblogs.com/liqingwen/archive/2017/05/08/6816100.html
-Advertisement-
Play Games

《Effective C#》快速筆記 - C# 中的動態編程 靜態類型和動態類型各有所長,靜態類型能夠讓編譯器幫你找出更多的錯誤,因為編譯器能夠在編譯時進行大部分的檢查工作。C# 是一種靜態類型的語言,不過它加入了動態類型的語言特性,可以更高效地解決問題。 本系列 《Effective C#》快速筆 ...


《Effective C#》快速筆記 - C# 中的動態編程

 

  靜態類型和動態類型各有所長,靜態類型能夠讓編譯器幫你找出更多的錯誤,因為編譯器能夠在編譯時進行大部分的檢查工作。C# 是一種靜態類型的語言,不過它加入了動態類型的語言特性,可以更高效地解決問題。  

 

本系列

  《Effective C#》快速筆記(一)- C# 語言習慣

  《Effective C#》快速筆記(二)- .NET 資源托管

  《Effective C#》快速筆記(三)- 使用 C# 表達設計

  《Effective C#》快速筆記(四) - 使用框架

  《Effective C#》快速筆記(五) - C# 中的動態編程

 

一、目錄

  • 三十八、理解動態類型的優劣
  • 三十九、使用動態類型表達泛型類型參數的運行時類型
  • 四十、將接受匿名類型的參數聲明為 dynamic
  • 四十一、用 DynamicObject 或 IDynamicMetaObjectProvider 實現數據驅動的動態類型
  • 四十二、如何使用表達式 API
  • 四十三、使用表達式將延遲綁定轉換為預先綁定
  • 四十四、儘量減少在公有 API 中使用動態類型

 

三十八、理解動態類型的優劣

  1. C# 動態類型是為了讓靜態代碼能夠更加平滑地與其他使用動態類型的環境進行交互,而不是鼓勵在一般場景中使用 dynamic 進行動態編程。

  2. 只要對象在運行時包含成員,那麼即可正常使用。

  3. 若是一個操作數(包括 this)為動態類型,那麼返回結果也會是動態類型。不過,最後依然要轉換成靜態類型,以便被其它 C# 代碼所使用,以及被編譯器感知。

  4. 當你需要不知道具體類型的運行時解析方法的時候,動態類型是最佳的工具。如果你能在編譯期間明確類型,那麼可以使用 lambda 表達式和函數式編程來解決問題。

  5. 表達式樹,一種在運行時創建代碼的方法(下麵提供示例)。

  6. 大多數情況,可以使用 Lambda 表達式創建泛型 API,讓調用者自己動態定義所需要執行的代碼即可。

  7. 優先使用靜態類型,靜態類型比動態類型更高效,動態類型和在運行時創建表達式樹都會帶來性能上的影響,即便這點影響微不足道。

  8. 若你能控製程序中所有涉及的類型時,可以引入一個介面,而不是動態類型,即基於介面編程,並讓所有需要支持該介面行為的類型都實現該介面。通過 C# 類型系統可以減少代碼在運行時所產生的錯誤,編譯器也能夠生成更加高效的代碼。

  9. 動態類型做法的效率比純粹的靜態類型下降挺大,但實現的難度卻比解析表達式樹要簡單地挺多。

 

  這裡,我使用 3 種數字相加的方法,dynamic 動態、Func 委托以及使用表達式樹進行相加的區別:

        [TestMethod]
        public void Test()
        {
            var result1 = AddDynamic(2, 3);
            Console.WriteLine((int)result1);

            var result2 = AddFunc(3, 4, (x, y) => x + y);
            Console.WriteLine(result2);

            var result3 = AddExpressionTree(4, 5);
            Console.WriteLine(result3);
        }

        /// <summary>
        /// Add,動態
        /// </summary>
        /// <param name="a"></param>
        /// <param name="b"></param>
        /// <returns></returns>
        private dynamic AddDynamic(dynamic a, dynamic b)
        {
            return a + b;
        }

        /// <summary>
        /// Add,使用委托
        /// </summary>
        /// <typeparam name="T1"></typeparam>
        /// <typeparam name="T2"></typeparam>
        /// <typeparam name="TR"></typeparam>
        /// <param name="a"></param>
        /// <param name="b"></param>
        /// <param name="func"></param>
        /// <returns></returns>
        private TR AddFunc<T1, T2, TR>(T1 a, T2 b, Func<T1, T2, TR> func)
        {
            return func(a, b);
        }

        /// <summary>
        /// Add,使用表達式樹
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="a"></param>
        /// <param name="b"></param>
        /// <returns></returns>
        private T AddExpressionTree<T>(T a, T b)
        {
            ParameterExpression leftOperand = Expression.Parameter(typeof(T), "left");
            ParameterExpression rightOperand = Expression.Parameter(typeof(T), "right");
            BinaryExpression body = Expression.Add(leftOperand, rightOperand);
            Expression<Func<T, T, T>> adder = Expression.Lambda<Func<T, T, T>>(body, leftOperand, rightOperand);

            Func<T, T, T> theDelegate = adder.Compile();
            return theDelegate(a, b);
        }
    }

 

三十九、使用動態類型表達泛型類型參數的運行時類型

  1. System.Linq.Enumerable.Cast<T> 將序列中的對象轉換成 T,從而使得 LINQ 可以配合 IEnumerable 進行工作。

  2. Convert<T> 要比 Cast<T> 適用性更廣,但同時也會執行更多的工作。

 

四十、將接受匿名類型的參數聲明為 dynamic

  1. 不要過度使用動態類型,因為動態調用會增加系統的額外開銷,即便不大。

  2. 長遠來看,具體類型更易於維護,編譯器和類型系統也會為其提供更好的支持。

  3. 擴展方法不能基於動態對象定義。

 

四十一、用 DynamicObject 或 IDynamicMetaObjectProvider 實現數據驅動的動態類型

  1. 創建帶有動態功能的類型的最簡單的方法就是繼承 System.Dynamic.DynamicObject。若能直接繼承 DynamicObject,那麼創建動態類就會比較簡單。

  2. 實現 IDynamicMetaObjectProvider 就意味著需要實現方法 GetmetaObject()。

  3. 創建動態類型時首選繼承,如果必須使用其他基類,可以手工實現 IDynamicMetaObjectProvider 介面,雖然所有的動態類型都會帶來性能上的損失,但這種手工實現介面的方式所帶來的損失往往更大一些。

 

  這是一個實現動態類型模型的一個示例。除了需要繼承 DynamicObject,還需要重寫 TryGetMemebr() 和 TrySetMemebr()。

        [TestMethod]
        public void Test1()
        {
            dynamic propDynamic = new DynamicPropertyBag();
            propDynamic.Now = DateTime.Now;

            Console.WriteLine(propDynamic.Now);
        }

        /// <summary>
        /// 動態屬性綁定模型
        /// </summary>
        internal class DynamicPropertyBag : DynamicObject
        {
            private readonly Dictionary<string, object> _storage = new Dictionary<string, object>();

            /// <summary>
            /// 獲取屬性值
            /// </summary>
            /// <param name="binder"></param>
            /// <param name="result"></param>
            /// <returns></returns>
            public override bool TryGetMember(GetMemberBinder binder, out object result)
            {
                var key = binder.Name;

                if (_storage.ContainsKey(key))
                {
                    result = _storage[key];
                    return true;
                }

                result = null;
                return false;
            }

            /// <summary>
            /// 設置屬性值
            /// </summary>
            /// <param name="binder"></param>
            /// <param name="value"></param>
            /// <returns></returns>
            public override bool TrySetMember(SetMemberBinder binder, object value)
            {
                var key = binder.Name;

                try
                {
                    if (_storage.ContainsKey(key))
                    {
                        _storage[key] = value;
                    }
                    else
                    {
                        _storage.Add(key, value);
                    }

                    return true;
                }
                catch (Exception e)
                {
                    Console.WriteLine(e);
                    return false;
                }
            }

 

 

四十二、如何使用表達式 API

  1. 傳統的反射 API 可以用表達式和表達式樹進行更好的替代,表達式可以直接編譯為委托。

  2. 介面使我們可以得到一個更為清晰、也更具可維護性的系統,反射是一個很強大的晚期綁定機制(雖然效率有所降低),.NET 框架使用它來實現 Windows 控制項和 Web 控制項的數據綁定。

 

  這裡提供了一個示例:

        [TestMethod]
        public void Test2()
        {
            var result = Call<int, string>((x) => (x * 2).ToString("D"));

            Console.WriteLine(result);
        }

        /// <summary>
        /// 調用
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <typeparam name="TResult"></typeparam>
        /// <param name="op"></param>
        /// <returns></returns>
        private TResult Call<T, TResult>(Expression<Func<T, TResult>> op)
        {
            var exp = op.Body as MethodCallExpression;
            var result = default(TResult);

            if (exp == null)
            {
                return result;
            }

            var methodName = exp.Method.Name;
            var parameters = exp.Arguments.Select(ProcessArgument);

            Console.WriteLine($"方法名 {methodName}");

            foreach (var parameter in parameters)
            {
                Console.WriteLine("參數:");
                Console.WriteLine($"\t{parameter.Item1}:{parameter.Item2}");
            }

            return result;
        }

        /// <summary>
        /// 處理參數
        /// </summary>
        /// <param name="expression"></param>
        /// <returns></returns>
        private Tuple<Type, object> ProcessArgument(Expression expression)
        {
            object arg = default(object);
            LambdaExpression l = Expression.Lambda(Expression.Convert(expression, expression.Type));

            Type parmType = l.ReturnType;
            arg = l.Compile().DynamicInvoke();

            return Tuple.Create(parmType, arg);
        }

 

四十三、使用表達式將延遲綁定轉換為預先綁定

  1. 延遲綁定API要使用符號(symbol)信息來實現,而預先編譯好的 API 則無需這些信息,表達式 API 正是二者之間的橋梁。

  2. 延遲綁定常見於 Silverlight 和 WPF 中使用的屬性通知介面,通過實現 INotifyPropertyChanged 和 INotifyPropertyChanging 介面來實現屬性變更的預綁定。

 

四十四、儘量減少在公有 API 中使用動態類型

  1. 優先使用 C# 的靜態類型,並儘可能地降低動態類型的作用範圍。若是想一直使用動態特性,你應該直接選用一種動態語言,而非 C#。

  2. 若要在程式中使用動態特性,請儘量不要在公有介面中使用,這樣會將動態類型限制在一個單獨的對象(或類型)中。

 

 

 


【博主】反骨仔

【原文】http://www.cnblogs.com/liqingwen/p/6816100.html

【GitHub】https://github.com/liqingwen2015/XMind 可以下載 XMind

【參考】《Effective C#》

 【參考】http://blog.csdn.net/w174504744/article/details/50562109


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

-Advertisement-
Play Games
更多相關文章
  • Linux常用命令 使用PHP伺服器端腳本編程語言進行網站開發,需要在lamp環境下進行,Linux作為”四劍客”之一是有必要瞭解熟悉的,而Linux系統並不像windows操作系統那樣,以圖形化的界面展示給我們,我們只需要滑鼠鍵盤配合使用即可完成我們的需要. 因此Linux系統的特殊性使我們有必要 ...
  • find命令:用於在文件樹中查找文件,並作出相應的處理 1、find命令的格式; find pathname -options [-print -exec -ok ...] {}\ 2、find命令的參數; pathname: find命令所查找的目錄路徑。例如用“.”來表示當前目錄,用“/”來表示 ...
  • 功能說明:設置指令別名 語法:alias[別名]=[指令名稱] 詳解:1)不加參數,顯示已有的別名列表 2)設置別名:alias NAME='VALUE',使用NAME時相當於執行VALUE的內容 特別事項:1)定義的別名指針對當前Shell有效 2)要永久有效,需要定義配置文件 當前用戶:~/.b ...
  • 數組(http://www.cnblogs.com/afei-24/p/6738128.html)的大小是固定的。如果元素的個數是動態的,就應使用集合類。 列表(http://www.cnblogs.com/afei-24/p/6824791.html) 隊列(待寫) 棧(待寫) 鏈表(待寫) 有序 ...
  • .NET Framework為動態列表List提供泛型類List<T>。這個類實現了IList,ICollection,IEnumerable,IList<T>,ICollection<T>,IEnumerable<T>介面。1.創建列表 創建一個賽車手類,下麵的例子會用到: 調用預設的構造函數,就 ...
  • 一轉眼務農6年了,呆過大公司也去過小作坊,碼農的人生除了摳腚還是摳腚。在所有呆過的公司里,感覺項目沒有不延期的,真的是因為自己不努力嗎?也沒有呀!上班不怎麼聊QQ回家也很少看動作片,還搞過幾次通宵擼碼的。 以前總感覺是項目經理把工時估少了,後來自己也做過項目管理,按照以往的經歷估工時,做到最後還是會 ...
  • 1 public static class DataSetUtilities { 2 public static void SendDataSet(DataSet ds) { 3 if (ds == null) { 4 throw new ArgumentException("SendDataSet... ...
  • 什麼是架構,談下你項目中用的是什麼樣的架構?!! What?這個問題怎麼熟悉!!原來是在面試的時候經常面試官會問出這個問題。 架構是項目系統實施的整個思維過程。他不是一件容易的事。 需要架構師能夠充分把握業務需求,對行業及項目具有前瞻性的眼光,對抽象思維有精深的能力,應對變化有足夠的反應,這樣才... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...