C#之扩展方法详解

前言:

我们想要向一个类型中添加方法,可以通过以下两种方式:

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);
        }
    }
}

打印结果:

扩展方法扩展的不仅仅可以是引用类型的类,也可以是值类型的类,附带一个官方的例子

C#之扩展方法详解_第1张图片

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

你可能感兴趣的:(c#)