十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
ASP.NET MVC学前篇之扩展方法、链式编程
我们拥有10多年网页设计和网站建设经验,从网站策划到网站制作,我们的网页设计师为您提供的解决方案。为企业提供成都网站设计、成都做网站、微信开发、成都微信小程序、成都手机网站制作、HTML5、等业务。无论您有什么样的网站设计或者设计方案要求,我们都将富于创造性的提供专业设计服务并满足您的需求。
目的没有别的,就是介绍几点在ASP.NETMVC 用到C#语言特性,还有一些其他琐碎的知识点,强行的划分一个范围的话,只能说都跟MVC有关,有的是外围的知识,有的是包含在框架内的。 MVC学前篇字样?有噱头的成分也有真实的成分,所以工欲善其事,必先利其器。器是什么?基础嘛,虽然说MVC框架中涉及到的知识很多很多也不是我一篇两篇能说完的,我能做的就是知道多少就跟大家分享多少,当然了随着时间的推移会完善这个系列。
扩展方法是C# 3.0特性里的知识,它用在最多的地方是在Linq中,把众多的查询功能添加到了IEnumerable和IEnumerable
运用的实际场景:有个购物清单(购物车)对象,它包含有添加、***清单内物品的功能
商品对象,它包含 商品名称、商品价格两个属性
代码1-1
1 namespace BlogCase 2 public class Commodity 3 { 4 public string Name { get; set; } 5 public float Price { get; set; } 6 } 7 8 namespace BlogCase 9 ///10 /// 购物清单 11 /// 12 public class ShoppingList 13 { 14 private List_Commodities; 15 16 public List Commodities 17 { 18 get { return _Commodities; } 19 } 20 21 public ShoppingList() 22 { 23 _Commodities = new List (); 24 } 25 26 public bool AddCommodity(Commodity commodity) 27 { 28 _Commodities.Add(commodity); 29 return true; 30 } 31 32 public bool RemoveCommodity(Commodity commodity) 33 { 34 if (_Commodities.Contains(commodity)) 35 { 36 _Commodities.Remove(commodity); 37 return true; 38 } 39 else 40 { 41 return false; 42 } 43 } 44 }
然后在这个令人不安、惶恐、期待、兴奋的新需求来了,要求清单可以提供一个清单内部所有货品价格的合计而又不破坏对象结构.
对的。这是非常合理的需求,现在咒骂也于事无补,因为需求总是未知的。在焦急、无奈中曙光就降临了,它就是C# 3.0中的特性扩展方法。
来看代码1-2
代码1-2
1 using BlogCase; 2 using System.Linq; 3 4 namespace BlogCase.Extension 5 { 6 public static class ShoppingListExtension 7 { 8 public static float Total(this ShoppingList shoppintlist) 9 { 10 return shoppintlist.Commodities.Sum(commodity => commodity.Price); 11 } 12 } 13 }
这里要说的是ShoppingListExtension类型是一个静态类,里面定义了一个静态方法Total,方法的签名是ShoppingList类型的参数,唯一不同的是在ShoppingList类型前面多了个this关键字,这时对ShoppingList类型的扩展方法已经定义好了。来看下示例代码1-3的使用吧。
代码1-3
1 using BlogCase.Extension; 2 3 namespace BlogCase 4 { 5 class Program 6 { 7 static void Main(string[] args) 8 { 9 ShoppingList shoppinglistTest = new ShoppingList(); 10 shoppinglistTest.AddCommodity(new Commodity() { Name = "A", Price = 14.3f }); 11 shoppinglistTest.AddCommodity(new Commodity() { Name = "B", Price = 15 }); 12 shoppinglistTest.AddCommodity(new Commodity() { Name = "C", Price = 27.9f }); 13 shoppinglistTest.AddCommodity(new Commodity() { Name = "D", Price = 34.3f }); 14 Console.WriteLine(shoppinglistTest.Total().ToString()); 15 Console.ReadLine(); 16 } 17 } 18 }
这里需要注意的是需要引用扩展方法类ShoppingListExtension所在的命名空间,在VS开发环境中,扩展方法的图标也和普通方法的不一样。如图1
图1
运行一下代码1-3,结果如图2。
图2
上面的内容是对类型来做扩展,添加了个扩展方法,这样对象间的耦合还是比较大的,假使修改了ShoppingList的内部结构,那ShoppingListExtension那里也得做相对应的修改,这里就要提到为什么要面相抽象编程了,后续的篇幅会提到。
现在我们要做的就是把ShoppingListExtension中的依赖类型换位更高层次的类型,并且再添加个扩展方法,用于过滤一些数据。由此全部结合起来可以看到一个链式编程的模型,对于学习linq和一些其他知识,这都是一个好的铺垫。
看下代码2-1示例,这是修改后的ShoppingList类型。
代码2-1
1 namespace BlogCase 2 { 3 public class ShoppingList : IEnumerable, IEnumerator 4 { 5 private List _Commodities = new List (); 6 public void Add(Commodity commodity) 7 { 8 _Commodities.Add(commodity); 9 } 10 public IEnumerator GetEnumerator() 11 { 12 return this; 13 } 14 IEnumerator IEnumerable.GetEnumerator() 15 { 16 return this; 17 } 18 private int _index = -1; 19 public Commodity Current 20 { 21 get 22 { 23 return _Commodities[_index]; 24 } 25 } 26 public void Dispose() 27 { 28 29 } 30 object System.Collections.IEnumerator.Current 31 { 32 get 33 { 34 return Current; 35 } 36 } 37 public bool MoveNext() 38 { 39 if (_index < _Commodities.Count) 40 { 41 _index++; 42 } 43 return (!(_index == _Commodities.Count)); 44 } 45 public void Reset() 46 { 47 _index = -1; 48 } 49 } 50 }
重新定义了ShoppingList类型,让其实现IEnumerable
对ShoppingListExtension类型也稍作修改,添加了新的扩展方法,并且修改了之前的扩展方法,达到了消耦的目的。
代码2-2
1 using BlogCase; 2 using System.Linq; 3 namespace BlogCase.Extension 4 { 5 public static class ShoppingListExtension 6 { 7 public static float Total(this IEnumerableshoppintlist) 8 { 9 return shoppintlist.Sum(commodity => commodity.Price); 10 } 11 12 public static IEnumerable Filter(this IEnumerable shoppinglist, Func commodityFilter) 13 { 14 var commodities = shoppinglist.Where(commodityFilter); 15 return commodities; 16 } 17 } 18 }
修改后的ShoppingListExtension类中,Total扩展方法的针对的类型变换掉了,而在Filter扩展方法中把返回类型定义为IEnumerable
代码2-3
1 using BlogCase.Extension; 2 namespace BlogCase 3 { 4 class Program 5 { 6 static void Main(string[] args) 7 { 8 ShoppingList shoppinglistTest = new ShoppingList(); 9 shoppinglistTest.Add(new Commodity() { Name = "A", Price = 50.3f }); 10 shoppinglistTest.Add(new Commodity() { Name = "B", Price = 60 }); 11 shoppinglistTest.Add(new Commodity() { Name = "C", Price = 70.9f }); 12 shoppinglistTest.Add(new Commodity() { Name = "D", Price = 80.3f }); 13 14 Console.WriteLine(shoppinglistTest.Filter(commodity=>commodity.Price>58).Total().ToString()); 15 Console.ReadLine(); 16 } 17 } 18 }
运行结果如图3
图3
从shoppinglistTest变量调用扩展方法Filter时,传入了一个查询条件(货品加个大于58)这个扩展方法返回的就是上面说的IEnumerable
到这里简单的一个链式编程模型就出来了,有兴趣的朋友可以接着去深入的了解linq,当然在此之前看完我的后续文章很重要。