c# - 使用和不使用异步声明的 lambda 之间有区别吗

标签 c# asynchronous lambda async-await

lambda () => DoSomethingAsync() 之间有区别吗?和 async () => await DoSomethingAsync()当两者都输入为 Func<Task> 时?我们应该选择哪一个?什么时候选择?

这是一个简单的控制台应用

using System;
using System.Threading.Tasks;

namespace asyncDemo
{
    class Program
    {
        static void Main(string[] args)
        {
            var demo = new AsyncDemo();
            var task = demo.RunTheDemo();
            task.Wait();

            Console.ReadLine();
        }
    }

    public class AsyncDemo
    { 
        public async Task Runner(Func<Task> action)
        {
            Console.WriteLine(DateTime.Now.ToLongTimeString() + " Launching the action");
            await action();
        }

        private async Task DoSomethingAsync(string suffix)
        {
            await Task.Delay(2000);
            Console.WriteLine(DateTime.Now.ToLongTimeString() + " Done something, " + suffix);
        }

        public async Task RunTheDemo()
        {
            await Runner(() => DoSomethingAsync("no await"));
            await Runner(async () => await DoSomethingAsync("with await"));
        }
    }
}

输出是:

09:31:08 Launching the action
09:31:10 Done something, no await
09:31:10 Launching the action
09:31:12 Done something, with await

所以在RunTheDemo , 两次调用 await Runner(someLambda);似乎以相同的时间特性做同样的事情——两者都有正确的两秒延迟。

两条线路都有效,那么它们是否完全等同? () => DoSomethingAsync() 之间有什么区别?和 async () => await DoSomethingAsync()结构体?我们应该选择哪一个?什么时候选择?

这与“我应该在一般情况下使用 await”不是同一个问题,因为这里我们正在处理工作异步代码,lambda 类型为 Func<Task>在消费方法中正确等待。问题涉及如何声明这些 lambda 以及该声明的效果。

最佳答案

Is there a difference between lambdas declared with and without async

是的,有区别。一个是异步 lambda,另一个只是任务返回 lambda。

一个异步 lambda 被编译成一个状态机,而另一个则没有,因此异步 lambda 具有不同的异常语义,因为异常被封装在返回的任务中并且不能同步抛出。

这与常规方法中存在的差异完全相同。例如在这个异步方法之间:

async Task FooAsync()
{
    await DoSomethingAsync("with await");
}

还有这个任务返回方法:

Task FooAsync()
{
    return DoSomethingAsync("no await");
}

查看这些方法可以更清楚地显示差异,但因为 lambda 只是语法糖,实际上被编译成与这些方法行为相同的方法。

Which one should we prefer and when?

这真的取决于你的口味。使用 async 关键字会生成一个状态机,其性能不如简单地返回一个任务。但是,在某些情况下,异常语义可能会令人惊讶。

以这段代码为例:

Hamster hamster = null;
Func<Task> asyncAction = () => FooAsync(hamster.Name);

var task = asyncAction();
try
{
    await task;
}
catch
{
    // handle
}

try-catch block 是否会处理 NullReferenceException

不会因为调用asyncAction时同步抛出异常。但是,在这种情况下,异常被处理,因为它在返回的任务中被捕获并在等待该任务时重新抛出。

Func<Task> asyncAction = async () => await FooAsync(hamster.Name);

我个人对这些单行表达式 lambda 使用任务返回 lambda,因为它们通常非常简单。但是我的团队,在出现一些极其有害的错误之后,总是使用 asyncawait 关键字。

关于c# - 使用和不使用异步声明的 lambda 之间有区别吗,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37133191/

相关文章:

c# - 在面板调整大小时保持按钮位置

python - 如何从 Flask 服务器发送多个异步请求?

node.js - 可以在 Node 全局范围内使用await来加载数据库客户端吗?

java - 在 Java 中调用包含可选值的列表?

lambda - 收集后获取java8上的索引

c# - 有没有一种自动处理数组和强类型集合之间转换的好方法?

c# - MySQLX.GetSession异常: : 'Expected message id: 2. Received message id: 10'

c# - 为什么该实例的继承链中的析构函数按顺序从最派生到最不派生被调用?

java - CompletableFuture runAsync 与新线程

django - 使用 AWS Lambda 部署 Django API 应用程序