使用Func委托与Async方法

时间:2016-05-17 15:27:14

标签: c# asynchronous lambda

我正在尝试使用Func和Async方法。我收到了一个错误。

  

无法将异步lambda表达式转换为委托类型'Func<HttpResponseMesage>'。异步lambda表达式可能返回void,Task或Task<T>,其中任何一个都不能转换为'Func<HttpResponseMesage>'

下面是我的代码:

public async Task<HttpResponseMessage> CallAsyncMethod()
{
    Console.WriteLine("Calling Youtube");
    HttpClient client = new HttpClient();
    var response = await client.GetAsync("https://www.youtube.com/watch?v=_OBlgSz8sSM");
    Console.WriteLine("Got Response from youtube");
    return response;
}

static void Main(string[] args)
{
    Program p = new Program();
    Task<HttpResponseMessage> myTask = p.CallAsyncMethod();
    Func<HttpResponseMessage> myFun =async () => await myTask;
    Console.ReadLine();
}

4 个答案:

答案 0 :(得分:56)

如错误所示,异步方法返回TaskTask<T>void。因此,要实现这一目标,您可以:

Func<Task<HttpResponseMessage>> myFun = async () => await myTask;

答案 1 :(得分:3)

代码修复,例如:

static void Main(string[] args)
        {
            Program p = new Program();
            Task<HttpResponseMessage> myTask = p.CallAsyncMethod();
            Func<Task<HttpResponseMessage>> myFun = async () => await myTask;
            Console.ReadLine();
        }

答案 2 :(得分:3)

我通常采用的路径是让Main方法调用返回任务的Run()方法,并.Wait()上的Task来完成。

class Program
{
    public static async Task<HttpResponseMessage> CallAsyncMethod()
    {
        Console.WriteLine("Calling Youtube");
        HttpClient client = new HttpClient();
        var response = await client.GetAsync("https://www.youtube.com/watch?v=_OBlgSz8sSM");
        Console.WriteLine("Got Response from youtube");
        return response;
    }

    private static async Task Run()
    {
        HttpResponseMessage response = await CallAsyncMethod();
        Console.ReadLine();
    }

    static void Main(string[] args)
    {
        Run().Wait();
    }
}

这允许您的控制台应用程序的其余部分以完全异步/等待支持运行。由于控制台应用程序中没有任何UI线程,因此您不会因使用.Wait()而面临死锁的风险。

答案 3 :(得分:-1)

在Func内部运行任务,等待它并检查异常,然后返回结果。

Func<HttpResponseMessage> myFun = () => 
{
   var t = Task.Run(async () => await myTask);
   t.Wait();
   if (t.IsFaulted)
      throw t.Exception;
   return t.Result;
};