.NET 扩展方法 (二)
上一篇随笔 .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 扩展方法 (二)的更多相关文章
- jQuery扩展方法笔记
		
一.方式列表: 1.jQuery.extend(Object); // jQuery 本身的扩展方法 2.jQuery.fn.extend(Object); // jQuery 所选对象扩展方法 二. ...
 - .NET中那些所谓的新语法之二:匿名类、匿名方法与扩展方法
		
开篇:在上一篇中,我们了解了自动属性.隐式类型.自动初始化器等所谓的新语法,这一篇我们继续征程,看看匿名类.匿名方法以及常用的扩展方法.虽然,都是很常见的东西,但是未必我们都明白其中蕴含的奥妙.所以, ...
 - ASP.Net MVC开发基础学习笔记:二、HtmlHelper与扩展方法
		
一.一个功能强大的页面开发辅助类—HtmlHelper初步了解 1.1 有失必有得 在ASP.Net MVC中微软并没有提供类似服务器端控件那种开发方式,毕竟微软的MVC就是传统的请求处理响应的回归. ...
 - Jquery自定义扩展方法(二)--HTML日历控件
		
一.概述 研究了上节的Jquery自定义扩展方法,自己一直想做用jquery写一个小的插件,工作中也用到了用JQuery的日历插件,自己琢磨着去造个轮子--HTML5手机网页日历控件,废话不多说,先看 ...
 - artDialog学习之旅(二)之扩展方法详解
		
名称 描述 核心方法 art.dialog.top 获取artDialog可用最高层window对象.这与直接使用window.top不同,它能排除artDialog对象不存在已经或者顶层页面为框架集 ...
 - linux系统下php安装mbstring扩展的二种方法
		
.执行 复制代码代码如下: yum install php-mbstring 2. 修改php.ini (这一步非常重要, 部分lxadmin版本无法自动修改) 复制代码代码如下: echo ‘ext ...
 - 23.C#Queryable的扩展方法(十二章12.1-12.2)
		
今天要写的知识还真心有点绕呢,对于第一节的内容,其实是把原先在内存中的数据源,换成了从数据库中提取出来的数据.从代码的使用方式上是一样的,直接跳过,来看看IEnumerable和IQueryable的 ...
 - MVC分页控件之二,为IQueryable定义一个扩展方法,直接反回PagedList<T>结果集(转)
		
namespace Entity { public interface IPagedList { /// <summary> /// 记录数 /// </summary> in ...
 - c#扩展方法的理解(二:接口)
		
namespace ExtensionInterfaceMethod { class Program { static void Main(string[] args) { //使用接口变量来调用扩展 ...
 
随机推荐
- EF遇到的一些问题
			
环境:EntityFramework 版本号:4.1.0.0 问题一:“数据读取器与指定的“.......”不兼容.某个类型为“...”的成员在同名的数据读取器中没有对应的列.”. 使用方式:rep. ...
 - (新年快乐)ABP理论学习之本地化(2016第一篇)
			
返回总目录 本篇目录 应用语言 本地化资源 获取本地化文本 扩展本地化资源 最佳实践 应用语言 一个应用至少有一种UI语言,许多应用不止有一种语言.ABP为应用提供了一个灵活的本地化系统. 第一件事情 ...
 - 循序渐进做项目系列(3):迷你QQ篇(1)——实现客户端互相聊天
			
<循序渐进做项目系列迷你QQ篇>将陆续介绍客户端聊天,文件传输,加好友,群聊,包括语音聊天,视频聊天,远程桌面等等需求如何实现,感兴趣的朋友可以持续关注.考虑到某些需求较为复杂,本系列采用 ...
 - Box-sizing:小身材,大拳头!
			
国庆回来,很久没写博客了.一来是自己毫无时间,二是最近开发任务特别紧,三是节后综合症,脑子一片空白没有找到写作的原材料.今天,在加完班回来的22点,忙里偷闲,分享一下最近学到的一个小知识点如题.标题的 ...
 - Redis性能问题排查解决手册(七)
			
阅读目录: 性能相关的数据指标 内存使用率used_memory 命令处理总数total_commands_processed 延迟时间 内存碎片率 回收key 总结 性能相关的数据指标 通过Red ...
 - Module Zero之权限管理
			
返回<Module Zero学习目录> 概览介绍 角色权限 用户权限 概览介绍 Module-Zero实现了ABP授权系统的IPermissionChecker接口.这篇文章中,我们将会看 ...
 - 【原】Python 用例:二进制写入和读取文件内容
			
import pickle as p shoplistfile='shoplist.data' shoplist=['apple','carrot'] # because the dump opera ...
 - Atitit 边缘检测原理attilax总结
			
Atitit 边缘检测原理attilax总结 1. 边缘检测的概念1 1.1. 边缘检测的用途1 2. 边缘检测方法分类1 3. 边缘检测的基本方法2 3.1. Roberts边缘检测算子2 3.2. ...
 - 安装Oracle 12c精简客户端(不带数据库)
			
注:Oracle客户端向下兼容,故也可以连接11g的数据库 下载页面皆为:http://www.oracle.com/technetwork/topics/winsoft-085727.html ...
 - memset
			
函数原型: void *memset(void *s, int ch, size_t n); 函数解释:将s中当前位置后面的n个字节 (typedef unsigned int size_t )用 c ...