1.问题提出
有时候,我希望能够调用一个方法判断一个string类型是否是合法的邮箱地址字符串。一般的可以采取以下方法:
定义一个专门的静态工具类,名为StringFormatChecker,并在其中定义一个方法,名为CheckMailAddrFormat,用于测试该字符串是否是合法的邮箱地址格式。
public static StringFormatChecker
{
...
//用于测试该字符串是否是合法的邮箱地址格式
public staic bool CheckMailAddrFormat(string str);
...
}
调用时:
...
string str = "Leon.Hong";
bool isValid = StringFormatChecker.CheckMailAddrFormat(str );
//isValid返回false
...
但是这里有两个问题:
(1) 对于方法的使用者来说,他除了学习string类外,还增加了了解StringFormatChecker类的学习成本
(2) 这个方法不是一个具有良好内聚性的方法,因为被检查的资源来源于string,但是却在另一个类中完成这个过程。
2.利用.Net的扩展方法(extension method)来解决这个问题
首先给出代码
public static StringFormatChecker
{
...
//用于测试该字符串是否是合法的邮箱地址格式
public staic bool CheckMailAddrFormat(this string str);
...
}
从代码看,不同之处,仅是对形参加了一个this进行修饰,但是,在调用时,却可以通过如下形式:
...
string str = "Leon.Hong";
bool isValid = str.CheckMailAddrFormat(str);
//isValid返回false
...
这个其他静态类的方法貌似变成了一个string的实例方法!!!
COOL&SHOCKING!!!
3.原理简要分析
(1)在.Net编译器中,它首先检查string是否提供了一个CheckMailAddrFormat(string),如果存在这样的一个实例方法,那么编译器就会生成IL来调用它。
(2)如果没有,则继续检查是否任何一个静态类定义了一个名为CheckMailAddrFormat(string)的静态方法,它的第一个参数是和当前用于调用方法的那个表达式的类型匹配的一个类型,而且这个类型必须用this关键字标识。同时,它再生成相应的IL代码来调用。