我正在开发一个 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/