C#異步調用的應用實踐最經公司工作需要調用一個外部的webservice,同時要將傳出的數據進行保存,以自己以前的習慣,就打算逐步操作,失敗啊,完全沒考慮過用戶體驗效果,在同事指點下,意識到使用C#異步調用的好處,隨便將自己找的一些資料留以保存,以戒後誤!
我們要明確,爲什麼要進行C#異步調用回調?衆所周知,普通方法運行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數據庫,網絡傳輸等),都會導致方法阻塞,表現在界面上就是,程序卡或者死掉,界面元素不動了,不響應了。C#異步調用方法很好的解決了這些問題,C#異步調用執行某個方法,程序立即開闢一個新線程去運行你的方法,主線程包括界面就不會死掉了。C#異步調用如何開始,好理解,現在我們討論的是如何結束這個C#異步調用出來的新線程。
首先,C#異步調用出來的新線程,必須回收,不回收是浪費資源的可恥行爲,.NET也是不允許的,所以你別想鑽空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分爲2種情況:主動回收和被動回收(當然,這是我自己的理解,微軟可不是這麼說的),主動回收就是,你去監視那個線程,並且等待,當C#異步調用方法完成了,就把C#異步調用線程回收,焦點回歸主線程,實際上就是上篇文章《C#C#異步調用初步》的那種情況,BeginInvoke之後又EndInvoke,如果在EndInvoke的時候,該C#異步調用線程沒有完成操作,那麼整個程序,包括主線程,又在阻塞了,又會出現界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“C#異步調用回調”。
C#異步調用核心有二:
A、 用回調函數(本例中爲CallBackMethod),C#異步調用結束後,自動調用此回調函數。
B、 而不在主線程中手工等待C#異步調用結束,如上兩例中在主線程中調用EndInvoke。此種方法,是在回調函數中調用EndInvoke的。
C#異步調用回調的大概流程是這樣的:首先啓動C#異步調用,啓動參數加上C#異步調用結束時執行的方法,然後這個C#異步調用線程就不用管了,最後當這個C#異步調用線程自己完成工作了,就自動執行啓動參數裏的那個方法,這樣確實很省心,可是代碼寫起來,就很複雜了。
下面是蒐藏的代碼:
- //首先準備好,要進行C#異步調用的方法(能C#異步調用的,最好不多線程)
- private string MethodName(int Num, out int Num2)
- {
- Num2 = Num;
- return "HelloWorld";
- }
- //程序終點
- //C#異步調用完成時,執行的方法(回調方法),
- //此方法只能有IAsyncResult一個參數,但是該參數幾乎萬能,可以傳遞object
- private void CallBackMethod(IAsyncResult ar)
- {
- //從C#異步調用狀態ar.AsyncState中,獲取委託對象
- DelegateName dn = (DelegateName)ar.AsyncState;
- //輸出參數
- int i;
- //一定要EndInvoke,否則你的下場很慘
- string r = dn.EndInvoke(out i, ar);
- MessageBox.Show("C#異步調用完成嘍!i的值是" i.ToString() ",r的值是" r);
- }
- //定義與方法同簽名的委託
- private delegate string DelegateName(int Num, out int Num2);
- //程序入口
- private void Run()
- {
- //實例化委託並初賦值
- DelegateName dn = new DelegateName(MethodName);
- //輸出參數
- int i;
- //實例化回調方法
- //把AsyncCallback看成Delegate你就懂了,
- //實際上AsyncCallback是一種特殊的Delegate,就像Event似的
- AsyncCallback acb = new AsyncCallback(CallBackMethod);
- //C#異步調用開始
- //如果參數acb換成null則表示沒有回調方法
- //最後一個參數dn的地方,可以換成任意對象,
- //該對象可以被回調方法從參數中獲取出來,寫成null也可以。
- //參數dn相當於該線程的ID,如果有多個C#異步調用線程,
- //可以都是null,但是絕對不能一樣,不能是同一個object,否則異常
- IAsyncResult iar = dn.BeginInvoke(1, out i, acb, dn);
- //去做別的事
- //…………
- }
- //最後的結果應該是:i=1,r="HelloWorld"
另外,如果可以,定義委託的時候可以選擇不用過多的修飾:
- /// ﹤summary﹥
- /// 定義委託
- /// ﹤/summary﹥
- /// ﹤returns﹥﹤/returns﹥
- public delegate bool Asyncdelegate();
- /// ﹤summary﹥
- /// Callback method must have the same signature as the
- /// AsyncCallback delegate
- /// ﹤/summary﹥
- /// ﹤param name="ar"﹥﹤/param﹥
- private void CallbackMethod(IAsyncResult ar)
- {
- // Retrieve the delegate.
- Asyncdelegate dlgt = (Asyncdelegate)ar.AsyncState;
- // Call EndInvoke to retrieve the results.
- dlgt.EndInvoke(ar);
- }
其他方法中調用:
- //C#異步調用執行
- //指定委託方法
- Asyncdelegate isgt = new Asyncdelegate(icpInfo.Insert);
- IAsyncResult ar = isgt.BeginInvoke(
- new AsyncCallback(CallbackMethod), isgt);
我的完整例子,保存。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
namespace ConsoleApplication1
{
public delegate void MyEventDelegate(int value);
public class Publisher
{
public MyEventDelegate myEvent;
}
public class Subscriber {
public void myMethod(int value)
{
Console.WriteLine(value);
}
}
class Program
{
static void Main(string[] args)
{
Publisher p = new Publisher();
Subscriber s1 = new Subscriber();
Subscriber s2 = new Subscriber();
// p.myEvent = new MyEventDelegate(s1.myMethod);
p.myEvent = s1.myMethod;
// p.myEvent += s2.myMethod;
//p.myEvent(100);
//p.myEvent.Invoke(100);
AsyncCallback acb = new AsyncCallback(CallBackMethod);
p.myEvent.BeginInvoke(100, acb, p.myEvent);
Console.Read();
}
private static void CallBackMethod(IAsyncResult ar)
{
//從C#異步調用狀態ar.AsyncState中,獲取委託對象
MyEventDelegate dn = (MyEventDelegate)ar.AsyncState;
//Console.WriteLine("C#異步調用完成嘍!i的值是{0}", ar.AsyncState.ToString());
//一定要EndInvoke,否則你的下場很慘
dn.EndInvoke(ar);
Console.WriteLine("C#異步調用完成嘍!i的值是{0}",ar.AsyncState.ToString());
}
}
}