c# - 为什么异步方法调用者的行为很奇怪?

标签 c# asynchronous async-await

我正在开发一个 Windows 8.1 商店应用程序,用户可以在其中将文本保存到文件中。

我一直在尝试了解如何最好地使用异步和等待。

这是我想出的:

private async void userText_KeyDown(object sender, KeyRoutedEventArgs e)
    {            
        if (e.Key == Windows.System.VirtualKey.Enter)
        {
            if (addUserImput)
            {
                userStringlist.Add(userBox.Text);
                userBox.Text = "";
                addUserImput = false;
            }
            await WriteToFileAsync();
            addUserImput = true;
        }
    }

异步方法如下所示:

private async Task WriteToFileAsync()
    {
        string name = "userStrings.txt";
        var option = CreationCollisionOption.ReplaceExisting;
        var folder = Windows.Storage.ApplicationData.Current.LocalFolder;
        var file = await folder.CreateFileAsync(name, option);
        await Windows.Storage.FileIO.WriteLinesAsync(file, userStringlist);            
    }

一旦 WriteToFileAsync 到达等待关键字,执行就会重新开始。为了防止列表中出现重复,我必须添加 if 语句。

这让我觉得很奇怪。我对此还很陌生,所以我可能错过了一些东西。为什么 keydown 事件从顶部恢复,执行已经完成的工作?

我的“解决方法”有效,我只是不明白事件行为背后的逻辑。

最佳答案

是的,这就是异步解决方案的工作原理。当您执行第一个实际上异步操作(在本例中为CreateFileAsync)时,该方法返回到其调用者,该调用者又返回到其调用者,并且最终会以正常方式工作。整个方法并备份到应用程序的消息循环。然后它继续处理其他 UI 消息。其中一些消息可能是按键事件(并且它们可能最终在异步操作完成之前运行)。其他事件可能是绘画事件或鼠标单击事件,它们可以让您的表单执行与用户交互所需的任何操作。这就是防止它卡住的原因。

您想要做的是防止给定的代码部分同时运行。如果这不是异步的,您通常可以使用 lock 关键字来解决这个问题,但这不是异步方法的一个选项。您需要的是某种方法来阻止访问代码,直到该代码块的任何其他执行完成。幸运的是,有一些工具可以做到这一点。您可以使用 bool 值,就像您一样,但这有点脆弱,而且随着应用程序复杂性的增长,很容易犯错误。信号量是专门为此任务设计的:

private SemaphoreSlim semaphore = new SemaphoreSlim(1);
private async void Bar()
{
    try
    {
        await semaphore.WaitAsync();
        //do stuff
    }
    finally
    {
        semaphore.Release();
    }
}

SemaphoreSlim 类有一个 WaitAsync 方法,专门设计用于异步方法,例如您的方法。您可以等到信号量空闲后再执行代码,然后确保完成后释放信号量,以便其他代码可以移入代码块。

关于c# - 为什么异步方法调用者的行为很奇怪?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22093459/

相关文章:

c# - Java 和 C# 中的 SQLite 实现

c# - 我怎样才能等待最短时间?

actionscript-3 - 我应该如何在访问器中执行异步操作?

javascript - 这个异步函数有什么问题吗?

c# - 为什么 WebClient.DownloadStringTaskAsync() 会阻塞? - 新的异步 API/语法/CTP

c# - 反射找不到嵌套类型的 protected 字段

c# - 在 Winforms TabControl 中动态创建 tabPages 的问题

javascript - 如何通过javascript从html页面中的特定github txt文件中获取数据

c# - 如何在 ASP MVC 中下载 NPOI 生成的 xls 文件

javascript - 如何重构检查图像(用作 css 背景法师)是否存在以返回 bool 值的函数?