前言:
我们想要向一个类型中添加方法,可以通过以下两种方式:
1.修改源代码。
2.在派生类中定义新的方法。
但是这两种方式都有缺点,1如果是别人的代码,你对其直接进行修改,可能破坏代码的完整性,使代码无法编译,2会增加代码的维护成本,修改功能时父类子类可能都要修改
1.扩展方法概述和基本准则
(1).C#只支持扩展方法,不支持扩展属性、扩展事件、扩展操作符等。
(2).扩展方法(第一个参数前面是this的方法)必须在非泛型的静态类中声明,扩展方法必须有一个参数,而且只有第一个参数使用this标记。
(3).用一个扩展方法扩展一个类型时,同时也扩展了派生类型。
2.扩展方法应用
namespace HelloWorld//命名空间
{
public class Student
{
public void FunctionA()
{
Console.WriteLine("我是方法A");
}
}
public static class Extension
{
public static void FunctionB(this Student myStudent, int a)
{
Console.WriteLine("我是Student类的扩展方法B,参数是" + a);
}
}
class ExtMethodDemo
{
static void Main(string[] args)
{
Student a=new Student();
a.FunctionB(5);
}
}
}
打印结果:
扩展方法扩展的不仅仅可以是引用类型,也可以是值类型,附带一个官方的例子
3.扩展方法本质
“扩展方法”是C#独有的一种方法,在扩展方法中会使用ExtensionAttribute这个attribute。
C#一旦使用this关键字标记了某个静态方法的第一个参数,编译器就会在内部向该方法应用一个定制的attribute,这个attribute会在最终生成的文件的元数据中持久性的存储下来,此属性在System.Core dll程序集中。
任何静态类只要包含了至少一个扩展方法,它的元数据中也会应用这个attribute,任何一个程序集包含了至少一个符合上述特点的静态类,它的元数据也会应用这个attribute。如果代码用了一个不存在的实例方法,编译器会快速的扫描引用的所有程序集,判断它们哪些包含了扩展方法,然后,在这个程序集中,可以扫描包含了扩展方法的静态类。
如果同一个命名空间中的两个类含有扩展类型相同的方法,就没有办法做到只用其中一个类中的扩展方法。为了通过类型的简单名称(没有命名空间前缀)来使用类型,可以导入该类型所有在的命名空间,但这样做的时候,你没有办法阻止那个命名空间中的扩展方法也被导入进来。
就比如说:
当编译看到以上述代码,编译器分两步工作:
(1) 编译器检查a当前类型,也就是Student类以及Student任何基类是否具有所匹配的名为FunctionB包含一个int参数的函数,如果找到,则生成IL代码并Call它;
(2) 如果没有找到匹配的方法,就继续检查是否有任何静态类定义了名为FunctionB的静态方法,并且这个方法必须第一个参数是用this关键字修饰,参数类型为Student的。找到时生成相应的IL代码来调用它。
所以也解释了为什么扩展类必须是静态非泛型的,因为你必须保证在编译阶段,Student能找到它匹配的方法,而泛型是在程序运行时才能确定方法。
官方文档链接:扩展方法 - C# 编程指南 - C# | Microsoft Learn
参考链接:https://www.cnblogs.com/yuer20180726/p/10901123.html