上一篇随笔 .NET 扩展方法 (一) 已经对 扩展方法有了大致的介绍,这篇算是一个补充,让我们来看一下扩展方法的几个细节:

一、扩展方法具有继承性

当使用扩展方法扩展一个类型的时候,其也扩展了派生类,所以上一篇的遗留问题“如果给object添加一个扩展方法会出现什么效果呢?” 的

答案就是——所有类型都将扩展该方法。object类已经经受住了时间的考验,我们似乎也找不到更合适的理由来扩展object类。从另外的

角度考虑,如果扩展了object类,很有可能会给“智能敏感提示”造成污染,以至于填充了过多的垃圾信息(因为许多类型也根本用不到该方法)。

二、扩展方法允许和扩展类型原有方法相等效的方法存在 (此处是个雷)

由于汉语语言表述的所带来的不易理解性,我们还是直接用代码来解释吧,如下的代码片段:

    public static class StringExtentsion
{
public static string ToString(this string str)
{
return "Extentsion" + str;
}
} class Program
{
static void Main(string[] args)
{
string str = "test"; Console.WriteLine(str.ToString()); // 输出结果为: test,也就说编译器会优先选用原有类的实例化方法,如果没找到匹配方法再寻找扩展方法 Console.Read();
}
}

由上述的代码片段可以知:StringExtentsion类中扩展方法ToString 和 String类的原有的ToString方法 对于客户端代码而言,它们的语法表象是

一样的,但本质上一个是StringExtentsion类的静态方法,一个是String类的实例化方法。然而编译运行没有产生错误,更没有产生警告。所以在

这种情况下很容“埋雷”,一不小心就会中招。有人也许会说:我注意一下不要和.NET类库的方法重名就可以了。但是你能保证 .NET 6、甚至.NET 10

的方法名和你写的绝对不重名吗?所以,扩展方法存在着版本控制的问题

 

三、扩展方法允许在两个或多个类中共存相同的扩展方法

    public static class StringExtentsion
{
public static bool IsEmpty(this string str)
{
return string.IsNullOrWhiteSpace(str);
}
} public static class OtherStringExtentsion
{
public static bool IsEmpty(this string str)
{
return string.IsNullOrWhiteSpace(str);
}
} class Program
{
static void Main(string[] args)
{
string str = null; //bool result = str.IsEmpty(); // 这种写法会产生编译错误:不明确的调用,因为它能找到两个扩展方法 bool result = StringExtentsion.IsEmpty(str); // 必须显示使用类调用静态方法 Console.WriteLine(result); Console.Read();
}
}

  

四、扩展其他类型的方法

扩展方法不仅可以扩展类类型,而且可以扩展接口、枚举、委托、结构、数组及对应的泛型类型 等类型。

在这里着重说一下扩展接口,任何 “实现了接口的类型对象” 都可以调用接口上的扩展方法。针对这一特点,我们完全可以将实现接口

的共用代码放进扩展方法里,实现代码复用。从形式上来看,该特点是 “单继承 + 接口” 与 “多继承”的中间产物,已经有了“多继承”的影子。

五、委托与扩展方法

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace MethodDemo
{
public static class StringExtentsion
{
public static string ShowString(this string str)
{
return "ShowString:" + str;
}
} class Program
{
static void Main(string[] args)
{
string str = "meng"; Func<string> fun = str.ShowString; fun(); Func<string> fun2 = str.ToString; fun2(); Console.Read();
}
}
}

  

如果你已经看过了 .NET 扩展方法 (一) ,也许你能猜到我接下来想说什么了吧。没错,看一下IL代码吧,看看编译器在背后搞了哪些“怪”。

顺便借助这个机会,说几个IL指令。

 .method private hidebysig static void  Main(string[] args) cil managed
{
.entrypoint
// 代码大小 55 (0x37)
.maxstack // 初始化3个局部变量
.locals init ([] string str,
[] class [mscorlib]System.Func`<string> fun,
[] class [mscorlib]System.Func`<string> fun2)
IL_0000: nop //将 "meng" 这个字符串对象的引用 入栈
IL_0001: ldstr "meng" // 将栈顶的值赋值给第0个局部变量(即 str),栈顶值出栈
IL_0006: stloc. // 将第0个局部变量入栈 (即 str 入栈)
IL_0007: ldloc. // 将 MethodDemo.StringExtentsion类的静态方法ShowString的指针入栈
IL_0008: ldftn string MethodDemo.StringExtentsion::ShowString(string) // 调用构造函数 new一个 Func<String>类型的委托
IL_000e: newobj instance void class [mscorlib]System.Func`<string>::.ctor(object,
native int)
IL_0013: stloc.
IL_0014: ldloc. // 调用fun对象的Invoke方法
IL_0015: callvirt instance ! class [mscorlib]System.Func`<string>::Invoke()
IL_001a: pop
IL_001b: ldloc.
IL_001c: dup // 将str 对象的实例化方法ToString方法的指针入栈
IL_001d: ldvirtftn instance string [mscorlib]System.Object::ToString()
IL_0023: newobj instance void class [mscorlib]System.Func`<string>::.ctor(object,
native int)
IL_0028: stloc.
IL_0029: ldloc.
IL_002a: callvirt instance ! class [mscorlib]System.Func`<string>::Invoke()
IL_002f: pop
IL_0030: call int32 [mscorlib]System.Console::Read()
IL_0035: pop
IL_0036: ret
} // end of method Program::Main

核心的IL指令已经给予了差不多的注释,根据IL指令可以得出结果:我们又被编译器“欺骗”了一次,fun对象保存的方法指针是

MethodDemo.StringExtentsion类的静态方法ShowString的指针。fun2对象保存的方法的指针str对象的ToString方法的指针。

.NET 扩展方法 (二)的更多相关文章

  1. jQuery扩展方法笔记

    一.方式列表: 1.jQuery.extend(Object); // jQuery 本身的扩展方法 2.jQuery.fn.extend(Object); // jQuery 所选对象扩展方法 二. ...

  2. .NET中那些所谓的新语法之二:匿名类、匿名方法与扩展方法

    开篇:在上一篇中,我们了解了自动属性.隐式类型.自动初始化器等所谓的新语法,这一篇我们继续征程,看看匿名类.匿名方法以及常用的扩展方法.虽然,都是很常见的东西,但是未必我们都明白其中蕴含的奥妙.所以, ...

  3. ASP.Net MVC开发基础学习笔记:二、HtmlHelper与扩展方法

    一.一个功能强大的页面开发辅助类—HtmlHelper初步了解 1.1 有失必有得 在ASP.Net MVC中微软并没有提供类似服务器端控件那种开发方式,毕竟微软的MVC就是传统的请求处理响应的回归. ...

  4. Jquery自定义扩展方法(二)--HTML日历控件

    一.概述 研究了上节的Jquery自定义扩展方法,自己一直想做用jquery写一个小的插件,工作中也用到了用JQuery的日历插件,自己琢磨着去造个轮子--HTML5手机网页日历控件,废话不多说,先看 ...

  5. artDialog学习之旅(二)之扩展方法详解

    名称 描述 核心方法 art.dialog.top 获取artDialog可用最高层window对象.这与直接使用window.top不同,它能排除artDialog对象不存在已经或者顶层页面为框架集 ...

  6. linux系统下php安装mbstring扩展的二种方法

    .执行 复制代码代码如下: yum install php-mbstring 2. 修改php.ini (这一步非常重要, 部分lxadmin版本无法自动修改) 复制代码代码如下: echo ‘ext ...

  7. 23.C#Queryable的扩展方法(十二章12.1-12.2)

    今天要写的知识还真心有点绕呢,对于第一节的内容,其实是把原先在内存中的数据源,换成了从数据库中提取出来的数据.从代码的使用方式上是一样的,直接跳过,来看看IEnumerable和IQueryable的 ...

  8. MVC分页控件之二,为IQueryable定义一个扩展方法,直接反回PagedList<T>结果集(转)

    namespace Entity { public interface IPagedList { /// <summary> /// 记录数 /// </summary> in ...

  9. c#扩展方法的理解(二:接口)

    namespace ExtensionInterfaceMethod { class Program { static void Main(string[] args) { //使用接口变量来调用扩展 ...

随机推荐

  1. 嵌入式的重要平台 .NET Micro Framework

    曾经辉煌的巨人PC界渐渐走向下坡路,而智能手机圈则没完没了般地争个你死我活.随着智能手机的广泛普及,不少商家为了不坐以待毙而纷纷开始涉足与穿戴式设备--智能手表(具体参见智能手表时代还有多远). 我们 ...

  2. ABP理论学习之Javascript API(理论完结篇)

    返回总目录 本篇目录 Ajax Notification Message UI block和busy 事件总线 Logging 其他工具功能 说在前面的话 不知不觉,我们送走了2015,同时迎来了20 ...

  3. 谈谈UI架构设计的演化

    谈谈UI架构设计的演化 经典MVC 在1979年,经典MVC模式被提出. 在当时,人们一直试图将纯粹描述思维中的对象与跟计算机环境打交道的代码隔离开来,而Trygve Reenskaug在跟一些人的讨 ...

  4. 老司机学新平台 - Xamarin开发之我的第一个MvvmCross跨平台插件:SimpleAudioPlayer

    大家好,老司机学Xamarin系列又来啦!上一篇MvvmCross插件精选文末提到,Xamarin平台下,一直没找到一个可用的跨平台AudioPlayer插件.那就自力更生,让我们就自己来写一个吧! ...

  5. Jquery UI - DatePicker 在Dialog中无法自动隐藏的解决思路

    通过Jquery UI Dialog模态展示如下的一个员工编辑页面,但是遇到一个奇怪的问题:点击Start Date的input元素后,其无法失去焦点.从而导致DatePicker控件在选择日期后无法 ...

  6. EMC学习之电磁辐射

    我们在接触新鲜事物的时候,通常习惯用自己熟悉的知识去解释自己不熟悉的事物.EMC知识更多的涉及到微波和射频,对于像我这种专注于信号完整性而 对EMC知识知之甚少的菜鸟来说,最初也只能用SI的一些基础知 ...

  7. iOS-多线程基础

    进程与线程: 1>   一个应用程序对应一个进程,一个进程帮助程序占据一块存储空间 2>   要想在进程中执行任务,就必须开启线程,一条线程就代表一个任务 3>   一个进程中允许开 ...

  8. Execute SQL Task 参数和变量的映射

    Execute SQL Task能够执行带参数的SQL查询语句或存储过程(SP),通过SSIS的变量(Variable)对参数赋值.对于不同的Connection Manager,在Task中需要使用 ...

  9. 希尔排序及希尔排序java代码

    原文链接:http://www.orlion.ga/193/ 由上图可看到希尔排序先约定一个间隔(图中是4),然后对0.4.8这个三个位置的数据进行插入排序,然后向右移一位对位置1.5.9进行插入排序 ...

  10. Caffe学习笔记1--Ubuntu 14.04 64bit caffe安装

    本篇博客主要用于记录Ubuntu 14.04 64bit操作系统搭建caffe环境,目前针对的的是CPU版本: 1.安装依赖库 sudo apt-get install libprotobuf-dev ...