AutoResetEvent配合webClient.DownloadDataAsync 接收不到信号问题

erictang2003 2013-12-19 02:40:54
就是很简单的从书上抄下来的,放在wpf的 MainWindow 中测试,但是发现AutoResetEvent 接收不到 .Set() 信号,永远都是超时webClient.CancelAsync() 的情况,求测试求解答。

public MainWindow()
{
InitializeComponent();

///////////////////////////////////////////////////////

AutoResetEvent downloadWaitEvent = new AutoResetEvent(false);
WebClient webClient = new WebClient();

webClient.DownloadDataCompleted += new DownloadDataCompletedEventHandler(rr);


var imgUrl = "http://img4.cache.netease.com/m/2013/12/18/20131218104403bd5df.png";//

//
webClient.DownloadDataAsync(new Uri(imgUrl), downloadWaitEvent);

if (downloadWaitEvent.WaitOne(10000)) //
{
//
System.Windows.MessageBox.Show("ll");
//
}
else
{
webClient.CancelAsync();
System.Windows.MessageBox.Show("CancelAsync");
}
}

private void rr(object sender, DownloadDataCompletedEventArgs e)
{
//
var waiterTmp = (AutoResetEvent)e.UserState;
//
try
{
//
if (!e.Cancelled && e.Error == null)
{
var bytes = e.Result;
//
}
}
finally
{
//
waiterTmp.Set(); //
//

}
}
...全文
150 5 打赏 收藏 转发到动态 举报
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
erictang2003 2014-01-22
  • 打赏
  • 举报
回复
经过跟踪,发现如果 webClient.DownloadDataAsync(new Uri(imgUrl)); downloadWaitEvent.WaitOne(10000) 这样写,实际webClient 仍然是使用和downloadWaitEvent.WaitOne 同一个线程,因此造成阻塞,实际异步下载没有执行。 ThreadPool.QueueUserWorkItem((state)=> { webClient.DownloadDataAsync(new Uri(imgUrl)); }, null); downloadWaitEvent.WaitOne(10000) 这样写就强制线程池给webClient.DownloadDataAsync 分配了另一个线程。
erictang2003 2014-01-22
  • 打赏
  • 举报
回复
进一步补充: webClient.DownloadDataAsyn 内部使用了 AsyncOperation asyncOperation = AsyncOperationManager.CreateOperation(userToken); 来返回异步操作, AsyncOperation 本身是对 System.Threading.SynchronizationContext 的包装, 它使用从 AsyncOperationManager.CreateOperation 调用中捕获的底层 SynchronizationContext 来以一种适合于创建时同步环境的方式回发委托。 例如,在 Windows® 窗体应用程序的 UI 线程上,SynchronizationContext.Current 可能会返回支持将委托调用封送回 UI 线程的 WindowsFormsSynchronizationContext。因此,如果在 UI 线程上调用 AsyncOperationManager.CreateOperation, 使用生成的 AsyncOperation 的 Post 方法会将提供的委托封送到 UI 线程并在其中执行。
erictang2003 2013-12-19
  • 打赏
  • 举报
回复
如果取消downloadWaitEvent.WaitOne(10000) 这句,一切都正常了,异步下载很快完成。 说明问题就在 AutoResetEvent 这里!
erictang2003 2013-12-19
  • 打赏
  • 举报
回复
如果if (downloadWaitEvent.WaitOne(10000)) 这句改为 if (downloadWaitEvent.WaitOne()) 那么会直接无限期等下去假死状态。 说明webClient.DownloadDataCompleted 根本就没有触发, 而如果WaitOne(10000) 那么总是在10秒到达时触发了webClient.DownloadDataCompleted 事件,但是 downloadWaitEvent.WaitOne(10000) 仍然返回 false,又引发 webClient.CancelAsync() 方法。 这段代码可是用微软msdn上直接复制下来的!居然是这样问题!求解答!

110,561

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧