本文参考自唔愛吃蘋果C#原始类型扩展方法—this参数修饰符,并在其基础上做了一些细节上的解释

1、this作为参数关键字的作用

使用this关键字,可以向this关键字后面的类型添加扩展方法,而无需给其创建新的派生类型、重新编译或以其他的方式修改类型.

扩展方法是一种特殊的静态方法,但可以像实例方法那样调用。但是this关键字最主要的作用是对类型的重载方法的扩充,来满足自身的需求,因为有些类方法的重载方法可能不是很完善,而直接修改类型的条件不足(应为可能有些已经被编译成dll,有些测试.net框架的底层类),所以只能通过this关键字对其进行扩展,也就是完善类方法的重载方法。

说了这么多,可能你还是不理解,现在有个需求:

我想利用string.format拼接一个,异常字符串,具体内容是:   ("parameter {0} cannot be empty",parameter),我知道使用string.format能很好地胜任,但是string.format静态方法我用不习惯,所以我想将它改造为string类的实例方法,直接通过"".实例方法来调用,最后的效果就是:"parameter {0} cannot be empty".实例方法名(parameter)

但是我不可能修改string类,来达到这个目地,因为string类是.Net Framework的底层类库,所以只能使用this关键字,来给其添加扩展方法,代码如下:

public static class StringUtilities
{
  public static string FormatWith(this string format,IFormatProvider provider,object org0) {
return format.FormatWith(provider, new object[] { org0 });
  }
  public static string FormatWith(this string format, IFormatProvider provider, object[] args) {
return string.Format(provider,format,args);
  }
}

ok,现在通过this关键字给string类添加了两个实例重载方法,但是需要注意,这里的类必须就静态类,扩展方法必须是静态方法,原因如下:

(1)、这里传入的需要扩展的string类型

不是静态变量而是成员变量

(2)、静态类的特点,如果一个静态类没有加构造函数,那么编译器会自动的给它加一个静态构造函数,静态构造函数是最早被调用的,只要有静态访问,那就先调用静态构造函数,接着调用非静态构造函数,而且静态类里面的成员在第一次被访问之后,就会被添加到全局环境中,后面的访问,将不会执行初始化操作,直接调用即可。而且静态类中不能有成员变量,this关键字是个列外

(3)、当类是静态类时,程序会在编译的时候,就将所有的静态成员编译到全局环境中,当类不是静态类的时候,只有当类中的静态成员被调用之后,才会被初始化到全局环境中,也就是说,代码如下:

public  class StringUtilities
{
public static string FormatWith(this string format, IFormatProvider provider, object org0)
{
return format.FormatWith(provider, new object[] { org0 });
}
}

这里的StringUtilities不是静态类,所以只要当StringUtilities的FormatWith在被初始化之后,他才会被初始化到全局环境中,下一次调用就不需要初始化了直接去全局环境中取,所以这个时候

这样是调不到FormatWith方法的,应为此时的FormatWith方法还没有被初始化,所以编译器会报错

,但是如果将StringUtilities改为静态类,那么编译器就会在编译完后,就将所有的静态成员初始化到全局环境中,这样上面出错的代码就没问题了,代码如下:

    public static class StringUtilities
{ public static string FormatWith(this string format, IFormatProvider provider, object org0)
{
return format.FormatWith(provider, new object[] { org0 });
} public static string FormatWith(this string format, IFormatProvider provider, params object[] args)
{
DataValidate.ArgumentNotNull(format, "format");
return string.Format(provider, format, args);
}
}

2、ok,上面的代码完成了对string类的扩展,为其添加了两个扩展方法,下面通过代码来测试是否成功

 string result = "'{0}' cannot empty".FormatWith(CultureInfo.InvariantCulture, "aaa");
Response.Write(result);

输出:

ok,说明实例方法扩展成功

总结:上面的扩展方法的调用方式,看上去像是成员方法,但实际编译器会对this关键字做特殊处理,编译器生成的中间语言(IL)会将代码转换为对静态方法的调用,

因此,并未真正违反封装原则。实际上,扩展方法无法访问它们所扩展的类型中的私有变量,不信你可以试试在扩展方法中访问string的私有成员!!!

C# this关键字(给底层类库扩展成员方法)的更多相关文章

  1. 话说TP框架里的Vendor这目录是干什么用的啊?类库扩展thinkphp3.1版本

    类库扩展包括基类库扩展.应用类库扩展和第三方类库扩展,所有扩展类库不会自动加载,需要手动加载或者定义别名和配置自动加载(详细可以参考4.2.3类库导入和4.2.5自动加载). 1 基类库扩展 目前支持 ...

  2. Golang把所有包括底层类库,输出到stderr的内容, 重新定向到一个日志文件里面?

    不论应用是如何部署的,我们都期望能扑捉到应用的错误日志, 解决思路: 自己写代码处理异常拦截,甚至直接在main函数中写异常拦截. stderr重定向到某个文件里 使用 syscall.Dup2 第一 ...

  3. Java精通并发-自旋对于synchronized关键字的底层意义与价值分析以及互斥锁属性详解与Monitor对象特性解说【纯理论】

    自旋对于synchronized关键字的底层意义与价值分析: 对于synchronized关键字的底层意义和价值分析,下面用纯理论的方式来对它进行阐述,自旋这个概念就会应运而生,还是很重要的,下面阐述 ...

  4. C#-this关键字的功能之扩展方法

    目录 1. 简介 2. 简单实例 3. 细节说明 1. 简介 我们的方法都是与声明他的类的相关联(我们现在写的各个方法都是在类中定义,所以我们调用方法都是用该方法所属类的实体对象调用). 在C#3.0 ...

  5. jquery-qrcode客户端二维码生成类库扩展--融入自定义Logo图片

    年后换了部门,现在主要的职责就是在网上卖精油,似乎这个就是传说中的网络营销. 跟着公司的MM们也了解不了少关于网络营销的知识,间接的了解到马云和刘强东都是些怎样龌龊的人,尽管之前也这样认为. 淘宝就不 ...

  6. iOS中__block 关键字的底层实现原理

    在 <iOS面试题集锦(附答案)> 中有这样一道题目: 在block内如何修改block外部变量?(38题)答案如下: 默认情况下,在block中访问的外部变量是复制过去的,即:写操作不对 ...

  7. Java的 volatile关键字的底层实现原理

    我们知道volatile关键字的作用是保证变量在多线程之间的可见性,它是java.util.concurrent包的核心,没有volatile就没有这么多的并发类给我们使用.本文详细解读一下volat ...

  8. java synchronized关键字的底层实现

    每个对象都有一个锁(Monitor,监视器锁),class对象也有锁,如果synchronized关键字修饰同步代码块,通过反编译可以看到,其实是有个monitorenter和monitorexit指 ...

  9. ..net 3.5新特性之用this关键字为类添加扩展方法

    具体用法如下: public static class ClassHelper { //用this 声明将要吧这个方法附加到Student对象 public static bool CheckName ...

随机推荐

  1. swoole1.8.0+版本异步redis安装(本实例为swoole1.8.10版本)详解

    Swoole-1.8.0+版本增加了对异步Redis客户端的支持,基于redis官方提供的hiredis库实现.Swoole提供了__call魔术方法,来映射绝大部分Redis指令(本次安装实例为sw ...

  2. [毕业设计][期末作业]二手闲置小程序 免费信息发布系统功能源码(小程序+php后台管理)

    最近做了一个小程序,主要是二手闲置免费信息发布系统的功能,里面包括了登录,发布商品,商品管理,违规投诉,canva商品海报生成,分享等一些基础的功能,可以说代码都是自己辛辛苦苦写出来的.可作为毕业设计 ...

  3. testNG的高级用法 --DataProvider

    @DataProvider Method参数 数据提供者的第一个参数是java.lang.reflect.Method,TestNG传递这个将调用的测试方法.如果您希望数据提供者根据不同的测试方法返回 ...

  4. python3.7使用models.ForeignKey时一定要传入实参on_delete=models.CASCADE

    models.ForeignKey 模型中最重要的部分——以及模型中唯一需要的部分——是它定义的数据库字段列表.字段由类属性指定.注意不要选择与模型API冲突的字段名称,如清除.保存或删除. from ...

  5. UDP实现一个简易的聊天室 (Unity&&C#完成)

    效果展示(尚未完善) UDP User Data Protocol 用户数据报协议 概述 UDP是不连接的数据报模式.即传输数据之前源端和终端不建立连接.使用尽最大努力交付原则,即不保证可靠交付. 数 ...

  6. C# defualt关键字默认值用法

    默认值表达式生成类型的默认值. 默认值表达式在泛型类和泛型方法中非常有用. 使用泛型类和泛型方法时出现的一个问题是,如何在无法提前知道以下内容的情况下将默认值赋值给参数化类型 T: T 是引用类型还是 ...

  7. .Net Core下使用Ajax,并传送参数到controllers

    可以使用URL拼接方式方法传参 .cshtml部分 @section Scripts{ @{ await Html.RenderPartialAsync("_ValidationScript ...

  8. UWP开发入门(十)——通过继承来扩展ListView

    本篇之所以起这样一个名字,是因为重点并非如何自定义控件,不涉及创建CustomControl和UserControl使用的Template和XAML概念.而是通过继承的方法来扩展一个现有的类,在继承的 ...

  9. sql数据库 大小查询

    select * from sys.master_files where name='CODA_PRD_Catalog' 12416*8/1024=(m)

  10. 如何实现一个IOS网络监控组件

    此文由作者朱志强授权网易云社区发布. Mobile Application Monitor IOS组件设计技术分享 背景 应用程序性能管理Application Performance Managem ...