C# 2008 学习笔记 - 扩展函数

转自:http://www.cnblogs.com/sunrack/articles/1073759.html

一、介绍
使用扩展函数,可以为无法修改源代码的对象添加新的方法,或者强制让对象支持某些方法,这些方法看起来就是对象本来就有的功能。

二、限制条件
1、必须在static> static class MyExtensions
{
// This method allows any object to display the assembly
// it is defined in.
public static void DisplayDefiningAssembly(this object obj)
{
Console.WriteLine("{0} lives here:\n\t->{1}\n", obj.GetType().Name,
Assembly.GetAssembly(obj.GetType()));
}
// This method allows any integer to reverse its digits.
// For example, 56 would return 65.
public static int ReverseDigits(this int i)
{
// Translate int into a string, and then
// get all the characters.
char[] digits = i.ToString().ToCharArray();
// Now reverse items in the array.
Array.Reverse(digits);
// Put back into string.
string newDigits = new string(digits);
// Finally, return the modified string back as an int.
return int.Parse(newDigits);
}
}
四、使用对象实例调用:

static  void Main( string[] args)
{
Console.WriteLine( " ***** Fun with Extension Methods *****\n ");
//  The int has assumed a new identity!
int myInt =  12345678;
myInt.DisplayDefiningAssembly();
//  So has the DataSet!
System.Data.DataSet d =  new System.Data.DataSet();
d.DisplayDefiningAssembly();
//  And the SoundPlayer!
System.Media.SoundPlayer sp =  new System.Media.SoundPlayer();
sp.DisplayDefiningAssembly();
//  Use new integer functionality.
Console.WriteLine( " Value of myInt: {0} ", myInt);
Console.WriteLine( " Reversed digits of myInt: {0} ", myInt.ReverseDigits());
myInt.Foo();
myInt.Foo( " Ints that Foo? Who would have thought it! ");
bool b2 =  true;
//  Error! Booleans don't have the Foo() method!
//  b2.Foo();
Console.ReadLine();
}

五、使用静态类调用:

private  static  void Main( string[] args)
{
Console.WriteLine( " ***** Fun with Extension Methods *****\n ");
int myInt =  12345678;
MyExtensions.DisplayDefiningAssembly(myInt);
DataSet d =  new DataSet();
MyExtensions.DisplayDefiningAssembly(d);
SoundPlayer sp =  new SoundPlayer();
MyExtensions.DisplayDefiningAssembly(sp);
Console.WriteLine( " Value of myInt: {0} ", myInt);
Console.WriteLine( " Reversed digits of myInt: {0} ",
MyExtensions.ReverseDigits(myInt));
TesterUtilClass.Foo(myInt);
TesterUtilClass.Foo(myInt,  " Ints that Foo? Who would have thought it! ");
Console.ReadLine();
}



static  class TesterUtilClass
{
//  Every Int32 now has a Foo() method
public  static  void Foo( this  int i)
{ Console.WriteLine( " {0} called the Foo() method. ", i); }
//  which has been overloaded to take a string!
public  static  void Foo( this  int i,  string msg)
{ Console.WriteLine( " {0} called Foo() and told me: {1} ", i, msg); }
}

六、扩展函数的使用范围
扩展函数本质上是在被扩展的对象实例上可以调用的静态函数,不是继承,所以不同于普通的成员函数,扩展函数不能直接访问被扩展对象的成员。只能通过该对象的实例来访问。

public  class Car
{
public  int Speed;
public  int SpeedUp()
{
return ++Speed;
}
}



public  static  class CarExtensions
{
public  static  int SlowDown( this Car c)
{
//  Error! This method is not deriving from Car!
return --Speed;
}
}




public  static  class CarExtensions
{
public  static  int SlowDown( this Car c)
{
//  OK!
return --c.Speed;
}
}




static  void UseCar()
{
Car c =  new Car();
Console.WriteLine( " Speed: {0} ", c.SpeedUp());
Console.WriteLine( " Speed: {0} ", c.SlowDown());
}


七、引用扩展函数

必须引用定义扩展函数的命名空间,否则扩展函数不可用

八、智能提示

Visual studio 的智能提示将扩展函数标记为向下的蓝色箭头

九、创建扩展函数库
比较好的做法是,定一个类库,在其中实现扩展函数

十、扩展接口类型
//  Define a normal CLR interface in C#.
interface IBasicMath
{
int Add( int x,  int y);
}
//  Implementation of IBasicMath.
class MyCalc : IBasicMath
{
public  int Add( int x,  int y)
{
return x + y;
}
}

但是,如果只是如下扩展,将会发生错误

static  class MathExtensions
{
//  Extend IBasicMath with subtraction method?
public  static  int Subtract( this IBasicMath itf,
int x,  int y);
}

必须给出函数的实现

static  class MathExtensions
{
//  Extend IBasicMath this method and this
//  implementation.
public  static  int Subtract( this IBasicMath itf,
int x,  int y)
{
return x - y;
}
}

调用
static  void Main( string[] args)
{
Console.WriteLine( " ***** Extending an interface *****\n ");
//  Call IBasicMath members from MyCalc object.
MyCalc c =  new MyCalc();
Console.WriteLine( " 1 + 2 = {0} ", c.Add( 1,  2));
Console.WriteLine( " 1 - 2 = {0} ", c.Subtract( 1,  2));
//  Can also cast into IBasicMath to invoke extension.
Console.WriteLine( " 30 - 9 = {0} ",
((IBasicMath)c).Subtract( 30,  9));
//  This would NOT work!
//  IBasicMath itfBM = new IBasicMath();
//  itfBM.Subtract(10, 10);
Console.ReadLine();
}

扩展接口后,显然不能直接在接口上调用这些扩展函数,只能理解为,所有继承该接口的对象新增加了这些扩展函数功能,

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值