有很多方法都是异步,那么如何从异步转到同步?

在本文开始,我必须告诉大家,这个方法可能立即死锁,所以使用的时候需要满足下面的条件

使用的条件

  1. 异步转同步的线程不是 UI 线程

  2. 如果线程是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();
            });
        }

参见:使用 Task.Wait()?立刻死锁(deadlock) - walterlv


本文会经常更新,请阅读原文: https://dotnet-campus.github.io//post/win10-uwp-%E5%BC%82%E6%AD%A5%E8%BD%AC%E5%90%8C%E6%AD%A5.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 lindexi (包含链接: https://dotnet-campus.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系