扩展方法使你能够向现有类型“添加”方法,而无需创建新的派生类型、重新编译或以其他方式修改原始类型。 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样进行调用。 对于用 C# 和 Visual Basic 编写的客户端代码,调用扩展方法与调用在类型中实际定义的方法之间没有明显的差异。
最常见的扩展方法是 LINQ 标准查询运算符,它将查询功能添加到现有的 System.CollectionsIEnumerable 和 System.Collections.GenericIEnumerableT 类型。 若要使用标准查询运算符,请先使用
using System.Linq;
指令将它们置于范围中。然后,任何实现了 IEnumerableT 的类型看起来都具有 GroupBy、 OrderBy、 Average 等实例方法。 在 IEnumerableT 类型的实例(如 ListT 或 Array)后键入“dot”时(就是符号点"."),可以在 IntelliSense 语句(IntelliSense翻译为“智能感知/智能感应功能”。)完成中看到这些附加方法。-------------一种 Microsoft 技术,这种技术通过在光标悬停在函数上时显示类定义和注释,从而让您可以分析源代码。当您在 IDE 中键入函数名时,IntelliSense 还可以完成这些名称。
IntelliSense 会在您编辑时自动插入代码,包括结束标记、右大括号以及值两边的引号。例如,如果您键入一个开始标记(如 <p>),它将自动插入结束标记(在本例中为 </p>)。--------
下面的示例演示如何对一个整数数组调用标准查询运算符 OrderBy 方法。 括号里面的表达式是一个 lambda 表达式。 很多标准查询运算符采用 lambda 表达式作为参数,但这不是扩展方法的必要条件。
1 class ExtensionMethods2 2 { 3 4 static void Main() 5 { 6 int[] ints = { 10, 45, 15, 39, 21, 26 }; 7 var result = ints.OrderBy(g => g); 8 foreach (var i in result) 9 { 10 System.Console.Write(i + " "); 11 } 12 } 13 } 14 //Output: 10 15 21 26 39 45
扩展方法被定义为静态方法,但它们是通过实例方法语法进行调用的。 它们的第一个参数指定该方法作用于哪个类型,并且该参数以 this 修饰符为前缀。 仅当你使用 using 指令将命名空间显式导入到源代码中之后,扩展方法才位于范围中。
下面的示例演示为 SystemString 类定义的一个扩展方法。 请注意,它是在非嵌套的、非泛型静态类内部定义的:
1 namespace ExtensionMethods 2 { 3 public static class MyExtensions 4 { 5 public static int WordCount(this String str) 6 { 7 return str.Split(new char[] { ‘ ‘, ‘.‘, ‘?‘ }, 8 StringSplitOptions.RemoveEmptyEntries).Length; //string类Split()方法的重载 9 } 10 } 11 }
可使用此 using 指令将
WordCount
扩展方法置于范围中:
using ExtensionMethods;
而且,可以使用以下语法从应用程序中调用该扩展方法:
string s = "Hello Extension Methods";
int i = s.WordCount();
在代码中,可以使用实例方法语法调用该扩展方法。 但是,编译器生成的中间语言 (IL) 会将代码转换为对静态方法的调用。 因此,并未真正违反封装原则。 实际上,扩展方法无法访问它们所扩展的类型中的私有变量。-----------------静态类种方法都设置为public static类型。
通用准则:
通常,建议你只在不得已的情况下才实现扩展方法,并谨慎地实现。 只要有可能,必须扩展现有类型的客户端代码都应该通过创建从现有类型派生的新类型来达到这一目的。
在使用扩展方法来扩展你无法更改其源代码的类型时,你需要承受该类型实现中的更改会导致扩展方法失效的风险。
如果你确实为给定类型实现了扩展方法,请记住以下几点:
- 如果扩展方法与该类型中定义的方法具有相同的签名,则扩展方法永远不会被调用。
- 在命名空间级别将扩展方法置于范围中。 例如,如果你在一个名为 Extensions的命名空间中具有多个包含扩展方法的静态类,则这些扩展方法将全部由 using Extensions;指令置于范围中。
针对已实现的类库,不应为了避免程序集的版本号递增而使用扩展方法。 如果要向你拥有源代码的库中添加重要功能,应遵循适用于程序集版本控制的标准 .NET Framework 准则。
扩展方法(C# 编程指南)