原文地址,请阅读原文: https://blog.lindexi.com/post/win10-uwp-%E5%BC%82%E6%AD%A5%E8%BD%AC%E5%90%8C%E6%AD%A5.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验
有很多方法都是异步,那么如何从异步转到同步?
在本文开始,我必须告诉大家,这个方法可能立即死锁,所以使用的时候需要满足下面的条件
使用的条件
-
异步转同步的线程不是 UI 线程
-
如果线程是UI线程,那么异步方法不能在另外一个线程。
看到这里也许你会疑惑,为何异步方法可以不在另一个线程?实际上对于 IO 等的异步方法,都是没有创建线程,请看There Is No Thread
关于这条件是如何来的,请看使用 Task.Wait()?立刻死锁(deadlock) - walterlv
使用方法
可以使用的方法需要获得是否有返回值,返回值是否需要。
如果需要返回值,使用GetResults
如从文件夹获取文件:
StorageFolder folder = StorageFolder.GetFolderFromPathAsync("").GetResults();
这是同步方法,几乎不需要做什么修改
如果是没有返回值或不需要返回值的,请看下面代码
StorageFolder.GetFolderFromPathAsync("").AsTask().Wait();
假设一个方法是没返回的,可以使用Wait
Foo().Wait();
private async Task Foo()
通过这个方法就可以把异步方法转同步。
如果需要反过来,把同步转异步,可以使用 同步方法转异步
await Task.Run(() =>
{
写你的代码
});
使用Task.Wait
时需要小心死锁
不会出现死锁的代码
直接在UI使用Task.Run
private void Button_OnClick(object sender, RoutedEventArgs e)
{
var n = Task.Run(() =>
{
return 2;
}).Result;
}
使用Task.Delay
等待
private void Button_OnClick(object sender, RoutedEventArgs e)
{
Task.Delay(100).Wait();
}
即使使用方法,里面使用 io 也有可能死锁
private void Button_OnClick(object sender, RoutedEventArgs e)
{
DoAsync().Wait();
}
private async Task DoAsync()
{
// 调用这个方法在 10.0.17134 / 10.0.16299 概率的死锁
// 在 10.0.17763 基本就会死锁
await ApplicationData.Current.LocalFolder.CreateFileAsync("lin", CreationCollisionOption.ReplaceExisting);
}
会出现死锁的写法
在UI使用异步会创建线程的方法
private void Button_OnClick(object sender, RoutedEventArgs e)
{
DoAsync().Wait();
}
async Task DoAsync()
{
await Task.Run(() => { });
}
private void Button_OnClick(object sender, RoutedEventArgs e)
{
DoAsync().Wait();
}
async Task DoAsync()
{
await Task.Delay(100);
}
private void Button_OnClick(object sender, RoutedEventArgs e)
{
DoAsync().Wait();
}
private async Task DoAsync()
{
await Task.Run( () =>
{
ApplicationData.Current.LocalFolder.CreateFileAsync("123",
CreationCollisionOption.ReplaceExisting).GetResults();
});
}
本文会经常更新,请阅读原文: https://blog.lindexi.com/post/win10-uwp-%E5%BC%82%E6%AD%A5%E8%BD%AC%E5%90%8C%E6%AD%A5.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者前往 CSDN 关注我的主页
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。