C#基础概念之委托
时间:2011-04-21 来源:杨延成
委托是C#中非常重要的一个概念,并在C#中得到了丰富的应用,如事件,线程等。那什么是委托呢?具体来说,委托是一种引用方法的类型。一旦为委托分配了方法,委托将与该方法具有完全相同的行为。委托方法的使用可以像其他任何方法一样,具有参数和返回值。
委托具有以下特点:
-
委托类似于 C++ 函数指针,但它是类型安全的。
-
委托允许将方法作为参数进行传递。
-
委托可用于定义回调方法。
-
委托可以链接在一起;例如,可以对一个事件调用多个方法。
-
方法不需要与委托签名精确匹配。有关更多信息,请参见协变和逆变。
-
C# 2.0 版引入了匿名方法的概念,此类方法允许将代码块作为参数传递,以代替单独定义的方法。
在C#中使用委托分为三步:
1.定义委托:
//声明委托
public delegate void MyDel();
2.实例化委托:
TestDel t = new TestDel();
Console.WriteLine("-----以下是简单使用委托演示--------");
//t.MyMethod();
///实例化委托,用一个方法来进行实例化
///该方法签名要与委托签名一致
MyDel del = new MyDel(t.MyMethod);
3.调用委托:
///调用委托
del();
好了,其实委托的变化很复杂,但基本都会符合这么三个步骤,说过了,这些,再来看一下完整的代码
namespace DelegateDemo
{
//声明委托
public delegate void MyDel();
//声明带参的委托
public delegate void MyDel2(int num1, int num2);
//声明带有返值的委托
public delegate string MyDel3(string s);
//声明委托用于演示匿名方法
public delegate string ProcessString(string s);
class Program
{
static void Main(string[] args)
{
#region 委托演示
/*
TestDel t = new TestDel();
#region 简单实例化委托与调用委托
Console.WriteLine("-----以下是简单使用委托演示--------");
//t.MyMethod();
///实例化委托,用一个方法来进行实例化
///该方法签名要与委托签名一致
MyDel del = new MyDel(t.MyMethod);
///调用委托
del();
//C#2.0后可以这种方式实例化委托
MyDel del4 = t.MyMethod;
del4();
//用静态方法进行实例化
del4 = TestDel.MyStaticMethod;
del4();
//以下代码效果相同
//MyDel2 del2 = new MyDel2(t.MyMethod);
//del2(10, 20);
MyDel2 del2 = t.MyMethod;
del2(10, 20);
//MyDel3 del3 = new MyDel3(t.MyMethod);
//Console.WriteLine(del3("abc"));
#endregion
#region 匿名方法实例化委托
Console.WriteLine("-----以下是匿名方法演示--------");
//用匿名方法实例化委托
ProcessString p = delegate(string inputString) {
return inputString.ToUpper();
};
//通过委托调用匿名方法
Console.WriteLine(p("aaaa"));
#endregion
#region 委托多播演示
Console.WriteLine("-----以下是委托多播演示--------");
MyDel mydel1 = t.MyMethod;
MyDel mydel2 = t.MyMethod2;
MyDel mydel3 = TestDel.MyMethod3;
MyDel allMyDel = mydel1 + mydel2 + mydel3;
allMyDel();
allMyDel -= mydel3;
allMyDel();
#endregion
#region 委托作为参数演示
Console.WriteLine("-------以下是委托作为参数演示------");
MyDel3 paramMyDel3 = t.MyMethod;
TestDel.MyParamMethod("aaa", paramMyDel3);
#endregion
#region 委托作为返回值
Console.WriteLine("---以下是委托作为返回值演示------");
///returnMyDel指向t.MyReturnMethod()的返回值
MyDel3 returnMyDel = t.MyReturnMethod();
///returnMyDel指向t.MyMethod
//MyDel3 returnMyDel = t.MyMethod;
Console.WriteLine(returnMyDel("sssssssssssss"));
#endregion
*/
#endregion
//MyReturnDelegateTest my = new MyReturnDelegateTest();
//my.MyTest();
MyParamDelegateTest myParam = new MyParamDelegateTest();
myParam.AddBooks();
myParam.MyTest();
}
}
public class TestDel
{
#region 普通方法
public static void MyStaticMethod()
{
Console.WriteLine("My Static Method");
}
public void MyMethod()
{
Console.WriteLine("MyMethod");
}
public void MyMethod2()
{
Console.WriteLine("My Method 22222222222");
}
public static void MyMethod3()
{
Console.WriteLine("My Method 3333333333333");
}
public void MyMethod(int num1, int num2)
{
Console.WriteLine(num1+num2);
}
public string MyMethod(string s)
{
return s.ToUpper();
}
#endregion
/// <summary>
/// 委托作为方法参数
/// </summary>
/// <param name="s"></param>
/// <param name="del3"></param>
public static void MyParamMethod(string s, MyDel3 del3)
{
Console.WriteLine(del3(s));
}
/// <summary>
/// 委托作为返回值
/// </summary>
/// <param name="s"></param>
/// <returns></returns>
public MyDel3 MyReturnMethod()
{
///返回符合委托规范的方法
return MyMethod;
}
}
委托作为参数示例:
委托作为参数public class MyParamDelegateTest
{
BookDB bookDB = new BookDB();
public void AddBooks()
{
bookDB.AddBook(new Book() { BookID=1,BookName="C#",Price=123,IsPaperbook=true });
bookDB.AddBook(new Book() { BookID = 1, BookName = "C#", Price = 123, IsPaperbook = false });
bookDB.AddBook(new Book() { BookID = 2, BookName = "ASP.Net", Price = 12, IsPaperbook = true });
bookDB.AddBook(new Book() { BookID = 1, BookName = "ADO", Price = 23, IsPaperbook = false });
}
/// <summary>
/// 用来实例化委托
/// </summary>
/// <param name="b"></param>
public void TestProcessBook(Book b)
{
if (b.IsPaperbook)
{
Console.WriteLine(b.BookName);
}
}
double total = 0;
public void TotalPrice(Book b)
{
total += b.Price;
}
public void MyTest()
{
//ProcessBook p=TestProcessBook;
//ProcessBook p1=TotalPrice;
//ProcessBook p2=p+p1;
//把方法名做为参数进行传递
bookDB.PrintBook(TestProcessBook);
bookDB.PrintBook(TotalPrice);
Console.WriteLine(total);
}
}
public delegate void ProcessBook(Book b);
public class BookDB
{
public List<Book> books = new List<Book>();
public void AddBook(Book b)
{
books.Add(b);
}
public void PrintBook(ProcessBook process)
{
foreach (var book in books)
{
process(book);
}
}
}
public class Book
{
public int BookID { get; set; }
public string BookName { get; set; }
public double Price { get; set; }
public bool IsPaperbook { get; set; }
}
委托作为返回值:
委托作为返回值public delegate int MyReturnDelegate(int num1, int num2);
public class MyReturnDelegateTest
{
public void MyTest()
{
MyCalcuate myCalcuate = new MyCalcuate();
do
{
Console.WriteLine("请输入符号进行以计算( + - * /)");
string oper = Console.ReadLine();
Console.WriteLine("请输入操作数1");
string num1 = Console.ReadLine();
Console.WriteLine("请输入操作数2");
string num2 = Console.ReadLine();
MyReturnDelegate myReturn = myCalcuate.Calcuate(oper);
int result = myReturn(int.Parse(num1), int.Parse(num2));
Console.WriteLine(
string.Format("{0}{1}{2}={3}", num1,oper,num2, result));
Console.WriteLine("您还要继续吗?Y/N");
//string continueFlag = Console.ReadLine();
//if (continueFlag.ToUpper() == "N") break;
} while (Console.ReadLine().ToUpper()!="N");
}
}
public class MyCalcuate
{
public MyReturnDelegate Calcuate(string oper)
{
MyReturnDelegate myReturn = null;
switch (oper)
{
case "+":
myReturn = delegate(int num1, int num2) { return num1 + num2; };
break;
case "-":
myReturn = delegate(int num1, int num2) { return num1 - num2; };
break;
case "*":
myReturn = delegate(int num1, int num2) { return num1 * num2; };
break;
case "/":
myReturn = delegate(int num1, int num2) { return num1 / num2; };
break;
default:
break;
}
return myReturn;
}
}