在C#中,委托类型是一个类型安全的、面向对象的函数指针。当我们通过delegate关键字定义一个委托类型后,编译器会给委托类型生成三个方法:Invoke、BeginInvoke和EndInvoke。
例如对于下面委托类型,可以通过ILSpy查看编译器生成的三个方法。
private delegate int NumberAdd(int a, int b);
同步执行委托实例
在使用委托的应用中,最常见的就是通过Invoke()方法以同步方式执行委托实例。也就是说,调用委托的线程将会一直等待,直到委托调用完成。
下面看一个同步执行委托的例子,在numberAdd委托实例中,通过Sleep(3000)模拟了一个耗时的操作:
NumberAdd numberAdd = (a, b) => { Thread.Sleep(3000); Console.WriteLine("----> NumberAdd() on thread is {0}", Thread.CurrentThread.ManagedThreadId); Console.WriteLine("----> start to calc {0} + {1}",a,b); Console.WriteLine("----> test result is {0}",a+b); return a + b; }; Console.WriteLine("main thread (id: {0}) invoke numberAdd function", Thread.CurrentThread.ManagedThreadId); //int result = numberAdd(2, 5); int result = numberAdd.Invoke(2, 5); Console.WriteLine("main thread (id: {0}) get the result: {1}", Thread.CurrentThread.ManagedThreadId, result);
代码的输出为下,从结果中可以看出,主线程执行委托实例过程中将会被阻塞,直到委托实例执行完成,主线程才会继续执行。
在很多应用中,一个方法可能要执行很久,例如加载一个很大的文档,或者执行一个耗时的数据库操作。如果我们使用同步的方式执行方法,那么主线程会一直阻塞,直到这个方法执行完成,表现就是应用程序没有相应,影响用户体验。这时,就可以考虑通过委托的异步性进行方法调用。