.net擴展方法可以擴展很多類型,包括:基本數據類型、接口、類,等等。如果,需要擴展的類型包含私有成員,擴展方法如何運用這些私有成員呢?本篇逐一體驗,包括:
■ 擴展基本數據類型
■ 擴展接口
■ 擴展包含私有字段的類 使用反射獲取類的私有字段
■ 擴展一個類的私有嵌套類 通過反射
擴展方法有幾個必要前提:
● 擴展方法所在的類必須是靜態類
● 擴展方法本身必須是靜態方法
● 擴展方法參數中,對類型的擴展參數前必須加this關鍵字
擴展基本數據類型
針對DateTime類型寫一個擴展方法。
public static class CalculateAge { public static int Age(this DateTime date, DateTime birthDate) { int birthYear = birthDate.Year; int currentYear = DateTime.Now.Year; if (birthYear >= currentYear) { throw new Exception("請輸入正確的出生日期~~"); } else { return currentYear - birthYear - 1; } } }
客戶端調用。
class Program { static void Main(string[] args) { try { Console.WriteLine("請輸入您的出生年份"); DateTime d = Convert.ToDateTime(Console.ReadLine()); DateTime dateInstance = new DateTime(); int age = dateInstance.Age(d); Console.WriteLine("您當前的年齡是:{0}", age); Console.ReadKey(); } catch (Exception ex) { Console.WriteLine(ex.Message); } } }
擴展接口
有這樣的一個產品模型。
public class Product { public int Id { get; set; } public string Name { get; set; } }
接口提供獲取產品集合的方法。
public interface IProductService { IEnumerable<Product> GetProducts(); }
接口有2個實現類。
public class FoodProducts : IProductService { public IEnumerable<Product> GetProducts() { return new List<Product> { new Product(){Id = 1, Name = "餅干"}, new Product(){Id = 2, Name = "牛奶"} }; } } public class ElectronicProducts : IProductService { public IEnumerable<Product> GetProducts() { return new List<Product> { new Product(){Id = 3, Name = "電風扇"}, new Product(){Id = 4, Name = "空調"} }; } }
針對接口擴展方法。
public static class ProductServiceExtension { public static IEnumerable<Product> GetProductsById(this IProductService productService, int id) { return productService.GetProducts().Where(p => p.Id == id); } }
客戶端調用。
class Program { static void Main(string[] args) { IProductService productService = new FoodProducts(); Console.WriteLine("食物類別下總數量是;{0}", productService.GetProducts().Count()); try { Console.WriteLine("找到的產品名稱是:{0}", (productService.GetProductsById(1).SingleOrDefault()).Name); } catch (Exception ex) { Console.WriteLine(ex.Message); } Console.ReadKey(); } }
擴展包含私有字段的類 使用反射獲取類的私有字段
擴展一個類的時候,有時候會用到該類的私有字段,我們可以通過反射拿到類的私有字段。
有這樣的一個類,包含私有字段和公共方法。
{ private DateTime _currentTime; public void SetTime() { _currentTime = DateTime.Now; } public string GetMsg() { if (_currentTime.Hour < 12) { return "上午好~~"; } else { return "下午好~~"; } } }
我們希望擴展出一個顯示英文信息的問候。
public static class DisplayMessageExtensions { public static string GetLocalMsg(this DisplayMessage message, string country) { //通過反射拿到私有字段 var privateField = typeof (DisplayMessage).GetField("_currentTime", BindingFlags.Instance | BindingFlags.NonPublic); //獲取該私有字段的值 var currentDateTime = (DateTime)privateField.GetValue(message); if (country == "USA" && currentDateTime.Hour < 12) { return "Good Morning"; } else { return "Good Evening"; } } }
客戶端調用。
class Program { static void Main(string[] args) { DisplayMessage displayMessage = new DisplayMessage(); displayMessage.SetTime(); Console.WriteLine("來自中國的問候是:{0}", displayMessage.GetMsg()); Console.WriteLine("美國人怎麼問候?"); Console.WriteLine("來自美國的問候是:{0}", displayMessage.GetLocalMsg("USA")); Console.ReadKey(); } }
擴展一個類的私有嵌套類 通過反射
當一個類有嵌套私有類的時候,擴展該類的時候,有時候會用到該類的嵌套私有類,我們可以通過反射擴展私有嵌套類。
有這樣的一個ParentClass類,包含一個私有嵌套類ChildClass.
public class ParentClass { public string MessageFromParent() { return "from parent~~"; } private class ChildClass { public string MessageFromChild() { return "from child~"; } } }
現在要擴展這個私有嵌套類,為其添加一個轉換成大寫的方法,通過反射來完成。
public static class NestedClassExtension { public static string ToUppeerCaseParentMessage(this ParentClass parent) { return parent.MessageFromParent().ToUpper(); } public static string ToUpperCaseChildMessage(this object o) { var childUpper = ""; //通過反射獲取父類中的私有嵌套類 var privateClass = typeof (ParentClass).GetNestedType("ChildClass", BindingFlags.NonPublic); if (o.GetType() == privateClass) { //通過反射獲取嵌套私有類的方法 var callMethod = privateClass.GetMethod("MessageFromChild"); childUpper = (callMethod.Invoke(o, null) as string).ToUpper(); } return childUpper; } }
客戶端,首先通過反射獲取私有嵌套類的type類型,然後運用私有嵌套類的擴展方法。
try { ParentClass p = new ParentClass(); //通過反射獲取父類私有嵌套類 var privateClass = typeof (ParentClass).GetNestedType("ChildClass", BindingFlags.NonPublic); //通過反射創建父類私有嵌套類的實例 var c = Activator.CreateInstance(privateClass); //通過反射獲取父類私有嵌套類的方法 //var callMethod = privateClass.GetMethod("MessageFromChild"); Console.WriteLine(c.ToUpperCaseChildMessage()); } catch (Exception ex) { Console.WriteLine(ex.Message); } Console.ReadKey();
var mi=list.GetType().GetMethod("Cast");
我是先把他的代碼黏貼到控制台下運行,結果報錯找不到這個方法後來想想,對了:
接著,我馬不停蹄趕到MSDN上查閱擴展方法定義在哪個類中在System.Linq命名空間下的Enumerable靜態類中。
找到了,OK!既然這個方法對於List是擴展,那麼對於這個Enumerable類而言不等於是普通的靜態方法嗎?好,既然如此,我就直接對你靜態類中的這個靜態方法進行反射了,出工了
List<string strings = new List<string { "1", "2", "3" };
MethodInfo mi = typeof(Enumerable).GetMethod("Cast", BindingFlags.Public | BindingFlags.Static);
var list = (IEnumerable<string)mi.Invoke(null, new object[]{strings}); //這裡出錯
哦,上網問問好友們,使用了一個方法:
var list = (IEnumerable<string)mi.
MakeGenericMethod(typeof(string)).
Invoke(null, new object[]{strings});
這個方法會將方法的泛型傳入反射裝置。然後進行反射調用。
你為啥要把類名設置為Class呢,那後面的Class clazz申明的就是你所寫的這個Class,不錯才怪。改一個類名吧