[.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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...