如何运行可能同步或可能异步的代码
本文关键字:异步 代码 同步 何运行 运行 | 更新日期: 2023-09-27 18:35:49
我有一个带有一些异步方法的类,这些方法采用处理程序类来决定成功或失败时要做什么。
public class FooModel
{
private FooState fooState;
public async Task UpdateFooAsync(IHandler handler)
{
try
{
var newFooState = await Something();
fooState = newFooState;
handler.Success();
}
catch
{
handler.Error();
}
}
}
public interface IHandler
{
void Success();
void Error();
}
我可能会从代码中的几个不同位置调用 UpdateFooAsync。根据我从哪里调用它,我可能想做不同的事情。例如,一个同步示例:
public async void update_click(object sender, RoutedEventArgs e)
{
await UpdateFooAsync(new Handler1());
}
public class Handler1 : IHandler
{
public void Success()
{
Provider.Page.text = "hello";
}
public void Error() {}
}
但我可能还想在以下之后执行异步操作:
private async void updateAndThenSomething_click(object sender, RoutedEventArgs e)
{
await UpdateFooAsync(new Handler2());
}
// this will generate not implemented error obviously
public class Handler2 : IHandler
{
public async void SuccessAsync()
{
await Provider.OperationAsync();
}
public void Error() {}
}
但是FooModel不知道我接下来是要执行同步操作还是异步操作。它不应该知道。有没有办法有处理者。成功() 运行同步代码还是异步代码?我读了一些Stephen Toub的东西(链接),这似乎暗示我不应该将成功作为一项任务,然后在UpdateFooAsync尝试块中等待该任务。
我想另一种选择是完全放弃处理程序接口。但这需要大量的重构。上述情况在我的代码中经常发生,通常不仅仅是成功和错误。我想强迫某人使用我的代码来指定成功、错误或其他任何事情时会发生什么。这就是我使用该界面的原因。我想我也可以通过抛出不同类型的异常来做到这一点,但我不确定这也是一个好主意。
您可以声明处理程序接口以返回Task
,然后在同步情况下使用 Task.FromResult
返回。
public interface IHandler
{
Task Success();
void Error();
}
public class Handler1 : IHandler
{
public Task Success()
{
Provider.Page.text = "hello";
return Task.FromResult<object>(null);
}
public void Error() {}
}
public class Handler2 : IHandler
{
public async Task Success()
{
return Provider.OperationAsync();
}
public void Error() {}
}