如何运行可能同步或可能异步的代码

本文关键字:异步 代码 同步 何运行 运行 | 更新日期: 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() {}
}