先看一段同步代碼:
public int SumPageSizes(IList<Uri> uris) {
int total = 0;
foreach (var uri in uris) {
statusText.Text = string.Format("Found {0} bytes ...", total);
var data = new WebClient().DownloadData(uri);
total += data.Length;
}
statusText.Text = string.Format("Found {0} bytes total", total);
return total;
}
這段代碼比較簡(jiǎn)單,使用同步方式一個(gè)一個(gè)的獲取Uri的Data,然后進(jìn)行統(tǒng)計(jì)。
如果要使用異步方式一個(gè)一個(gè)的統(tǒng)計(jì),那應(yīng)該如何計(jì)算呢?
我以前演示過(guò)一段丑陋的代碼大致如下:
WebClient webClient = new WebClient();
webClient.DownloadDataCompleted += (s, e) =>
{
// 使用A對(duì)象,做些事情。
WebClient webClient2 = new WebClient();
webClient2.DownloadDataCompleted += (s2, e2) =>
{
//使用B對(duì)象,做些事情。
// 遞歸的去 DownloadDataAsync。
};
webClient2.DownloadDataAsync(new Uri("B 的地址"));
};
webClient.DownloadDataAsync(new Uri("A 的地址"));
當(dāng)然如果你確定只有兩個(gè)地址的話(huà),這種方法未嘗不可。如果有多個(gè)地址的話(huà),則必須遞歸的調(diào)用了。
如何使用Enumerator來(lái)簡(jiǎn)化異步操作:
public void SumPageSizesAsync(IList<Uri> uris) {
SumPageSizesAsyncHelper(uris.GetEnumerator(), 0);
}
private void SumPageSizesAsyncHelper(IEnumerator<Uri> enumerator, int total) {
if (enumerator.MoveNext()) {
statusText.Text = string.Format("Found {0} bytes ...", total);
var client = new WebClient();
client.DownloadDataCompleted += (sender, e) => {
SumPageSizesAsyncHelper(enumerator, total + e.Result.Length);
};
client.DownloadDataAsync(enumerator.Current);
}
else {
statusText.Text = string.Format("Found {0} bytes total", total);
enumerator.Dispose();
}
}
通過(guò)SumPageSizesAsyncHelper ,可以實(shí)現(xiàn)異步調(diào)用A->異步調(diào)用B->異步調(diào)用C..的方式。
首先解釋下為什么可以,假設(shè)uris 有A,B,C.
SumPageSizesAsyncHelper(uris.GetEnumerator(), 0);
方法先調(diào)用A,因?yàn)锳后面還有B,所以enumerator.MoveNext()返回True,
接著在DownloadDataCompleted事件結(jié)束后,調(diào)用B,同樣,因?yàn)锽后面還有C,
所以enumerator.MoveNext() 繼續(xù)返回True,接著在DownloadDataCompleted事件后調(diào)用C。
在調(diào)用C結(jié)束后,因?yàn)镃后面沒(méi)有了,所以enumerator.MoveNext() 返回False,
也可以認(rèn)為全部都下載完畢了。所以返回最終的結(jié)果。
如果使用async 和await來(lái)實(shí)現(xiàn)的話(huà),代碼如下:
public async Task<int> SumPageSizesAsync2(IList<Uri> uris)
{
int total = 0;
Char charText = 'A';
foreach (var uri in uris)
{
var data = await new WebClient().DownloadDataTaskAsync(uri);
total += data.Length;
Console.WriteLine("Thread Id: {0}:調(diào)用{1}的地址 Found {2} bytes...{3}",
Thread.CurrentThread.ManagedThreadId, charText, total, DateTime.Now);
charText = Convert.ToChar(charText + 1);
}
Console.WriteLine("Thread Id: {0}:全部完成,Found {1} bytes total {2}",
Thread.CurrentThread.ManagedThreadId, total, DateTime.Now);
return total;
}
新聞熱點(diǎn)
疑難解答
圖片精選