解析C#中委托的同步調(diào)用與異步調(diào)用(實例詳解)
更新時間:2013年05月18日 11:59:56 作者:
本篇文章是對C#中委托的同步調(diào)用與異步調(diào)用進行了詳細的分析介紹,需要的朋友參考下
委托的Invoke方法用來進行同步調(diào)用。同步調(diào)用也可以叫阻塞調(diào)用,它將阻塞當前線程,然后執(zhí)行調(diào)用,調(diào)用完畢后再繼續(xù)向下進行。
同步調(diào)用的例子:
using System;
using System.Threading;
public delegate int AddHandler(int a, int b);
public class Foo {
static void Main() {
Console.WriteLine("**********SyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
int result = handler.Invoke(1,2);
Console.WriteLine("Do other work... ... ...");
Console.WriteLine(result);
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
}運行結(jié)果:
**********SyncInvokeTest**************
Computing 1 + 2 ...
Computing Complete.
Do other work... ... ...
同步調(diào)用會阻塞線程,如果是要調(diào)用一項繁重的工作(如大量IO操作),可能會讓程序停頓很長時間,造成糟糕
的用戶體驗,這時候異步調(diào)用就很有必要了。
異步調(diào)用不阻塞線程,而是把調(diào)用塞到線程池中,程序主線程或UI線程可以繼續(xù)執(zhí)行。
委托的異步調(diào)用通過BeginInvoke和EndInvoke來實現(xiàn)。
異步調(diào)用:
using System;
using System.Threading;
public delegate int AddHandler(int a, int b);
public class Foo {
static void Main() {
Console.WriteLine("**********AsyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
IAsyncResult result = handler.BeginInvoke(1,2,null,null);
Console.WriteLine("Do other work... ... ...");
Console.WriteLine(handler.EndInvoke(result));
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
}運行結(jié)果: **********AsyncInvokeTest**************
Do other work... ... ...
Computing 1 + 2 ...
Computing Complete.
可以看到,主線程并沒有等待,而是直接向下運行了。
但是問題依然存在,當主線程運行到EndInvoke時,如果這時調(diào)用沒有結(jié)束(這種情況很可能出現(xiàn)),這時為了等待調(diào)用結(jié)果,線程依舊會被阻塞。
解決的辦法是用回調(diào)函數(shù),當調(diào)用結(jié)束時會自動調(diào)用回調(diào)函數(shù)
回調(diào)異步:
public class Foo {
static void Main() {
Console.WriteLine("**********AsyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
IAsyncResult result = handler.BeginInvoke(1,2,new AsyncCallback(AddComplete),"AsycState:OK");
Console.WriteLine("Do other work... ... ...");
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
static void AddComplete(IAsyncResult result) {
AddHandler handler = (AddHandler)((AsyncResult)result).AsyncDelegate;
Console.WriteLine(handler.EndInvoke(result));
Console.WriteLine(result.AsyncState);
}
}
同步調(diào)用的例子:
復制代碼 代碼如下:
using System;
using System.Threading;
public delegate int AddHandler(int a, int b);
public class Foo {
static void Main() {
Console.WriteLine("**********SyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
int result = handler.Invoke(1,2);
Console.WriteLine("Do other work... ... ...");
Console.WriteLine(result);
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
}運行結(jié)果:
**********SyncInvokeTest**************
Computing 1 + 2 ...
Computing Complete.
Do other work... ... ...
同步調(diào)用會阻塞線程,如果是要調(diào)用一項繁重的工作(如大量IO操作),可能會讓程序停頓很長時間,造成糟糕
的用戶體驗,這時候異步調(diào)用就很有必要了。
異步調(diào)用不阻塞線程,而是把調(diào)用塞到線程池中,程序主線程或UI線程可以繼續(xù)執(zhí)行。
委托的異步調(diào)用通過BeginInvoke和EndInvoke來實現(xiàn)。
異步調(diào)用:
復制代碼 代碼如下:
using System;
using System.Threading;
public delegate int AddHandler(int a, int b);
public class Foo {
static void Main() {
Console.WriteLine("**********AsyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
IAsyncResult result = handler.BeginInvoke(1,2,null,null);
Console.WriteLine("Do other work... ... ...");
Console.WriteLine(handler.EndInvoke(result));
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
}運行結(jié)果: **********AsyncInvokeTest**************
Do other work... ... ...
Computing 1 + 2 ...
Computing Complete.
可以看到,主線程并沒有等待,而是直接向下運行了。
但是問題依然存在,當主線程運行到EndInvoke時,如果這時調(diào)用沒有結(jié)束(這種情況很可能出現(xiàn)),這時為了等待調(diào)用結(jié)果,線程依舊會被阻塞。
解決的辦法是用回調(diào)函數(shù),當調(diào)用結(jié)束時會自動調(diào)用回調(diào)函數(shù)
回調(diào)異步:
復制代碼 代碼如下:
public class Foo {
static void Main() {
Console.WriteLine("**********AsyncInvokeTest**************");
AddHandler handler = new AddHandler(Add);
IAsyncResult result = handler.BeginInvoke(1,2,new AsyncCallback(AddComplete),"AsycState:OK");
Console.WriteLine("Do other work... ... ...");
Console.ReadLine();
}
static int Add(int a, int b) {
Console.WriteLine("Computing "+a+" + "+b+" ...");
Thread.Sleep(3000);
Console.WriteLine("Computing Complete.");
return a+b;
}
static void AddComplete(IAsyncResult result) {
AddHandler handler = (AddHandler)((AsyncResult)result).AsyncDelegate;
Console.WriteLine(handler.EndInvoke(result));
Console.WriteLine(result.AsyncState);
}
}
相關文章
C#Windows窗體設計之ContextMenuStrip(鼠標右擊菜單)的使用
這篇文章主要介紹了C#Windows窗體設計之ContextMenuStrip(鼠標右擊菜單)的使用方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07C#基于WebBrowser獲取cookie的實現(xiàn)方法
這篇文章主要介紹了C#基于WebBrowser獲取cookie的實現(xiàn)方法,實例分析了C#基于WebBrowser簡單讀取瀏覽谷歌網(wǎng)站cookie的相關技巧,非常簡單實用,需要的朋友可以參考下2015-11-11asp.net core mvc權限控制:在視圖中控制操作權限
本文主要介紹了asp.net core mvc權限控制:在視圖中控制操作權限。具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02C# web.config之<customErrors>節(jié)點說明案例詳解
這篇文章主要介紹了C# web.config之<customErrors>節(jié)點說明案例詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08