繁体   English   中英

从Task获取对象并从非异步方法返回

[英]Get object from Task and return from non-async method

我有一个较低级的库,该库是我设计为异步并返回Task的方法之一。

我知道在较低级别的库中使用异步方法不是一个好主意,但是由于处理的繁琐,因此无法解决这一问题。

在我的低级库之间,我有一个主库,它公开了所涉及的低级库方法。 “就像一个业务层”。

我的应用程序的入口点是WEB Api,然后调用中级库,而中级则调用最低级库。

我的相关问题是,最好从最低层库中解压缩结果,在该最低层库中,异步链从我的中间库开始,然后向消费者公开一个静态方法,或者我应该只为消费者提供一个异步方法而仅向消费者提供?从最低到最高的异步方法?

从应用程序的入口开始,有问题的第一个方法是: CutSheet.CutSheet cutSheet = AlumCloudPlans.Manager.GetCutSheet(elev);

我知道必须有一个更好的方法,因为在中级库上,我必须创建一个全新的任务,并且返回值是value,这是一个Task,然后返回该Task的值。

您会建议我在中级库中做什么?

-最低水平

namespace CutSheet
{
    public class Manager
    {
        public async static Task<CutSheet> GetCutSheet(IElevation elevation)
        {
            return new CutSheet(await new PartsProcessor.PartProcessor()
                                                 .ProcessParts(elevation));
        }
    }
}

- -中级

namespace AlumCloudPlans
{
    public class Manager
    {    
        public static CutSheet.CutSheet GetCutSheet(IElevation elevation)
        {
            var t = Task.Factory.StartNew(async delegate
            {
                return await CutSheet.Manager.GetCutSheet(elevation);
            });

            t.Wait();
            return t.Result.Result;
        }

    }
}

---最高级别和进入应用程序

namespace StorefrontSystem.Controllers
{
    [Authorize]
    public class CutSheetController : AlumCloudWebApiBaseController
    {

        public async Task<HttpResponseMessage> Get(string a, string f, int id)
        {
            HttpResponseMessage r = Request.CreateResponse();

            IElevation elev = await ElevationManager.GetElevation(id);

            CutSheet.CutSheet cutSheet = AlumCloudPlans.Manager
                                                   .GetCutSheet(elev);

            var ms = new MemoryStream();
            bool isPDF = f.ToLower() == "pdf";

            if (isPDF)
            {
                using (var pdf = await cutSheet.GetCutSheetPDF(elev))
                {
                    pdf.Save(ms, false);
                }
            }
            else
            {    
                using (Bitmap canvas = await cutSheet.GetDrawing())
                {
                    canvas.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
                }    
            }

            ms.Position = 0;
            r.Content = new StreamContent(ms);
            if (isPDF)
            {
                r.Content.Headers.ContentType 
                      = new Headers.MediaTypeHeaderValue("Application/pdf");
            }
            else
            {
                r.Content.Headers.ContentType 
                     = new Headers.MediaTypeHeaderValue("image/png");
            }
            return r;
        }
    }
}

我知道在较低级别的库中使用异步方法不是一个好主意

谁告诉你的? 只要您正在执行的操作自然是异步的,那么在任何级别上都具有async方法是一个绝妙的主意。 但是,在较低级别中,通过使用Task.Run 假装某些异步对象并不是一个好主意-在ASP.NET中尤其如此。

假设您的操作自然是异步的,则应该一直使用async

中级:

public class Manager
{
    public static Task<CutSheet.CutSheet> GetCutSheetAsync(IElevation elevation)
    {
        return CutSheet.Manager.GetCutSheetAsync(elevation);
    }
}

最高级别:

[Authorize]
public class CutSheetController : AlumCloudWebApiBaseController
{
    public async Task<HttpResponseMessage> Get(string a, string f, int id)
    {
        HttpResponseMessage r = Request.CreateResponse();

        IElevation elev = await ElevationManager.GetElevationAsync(id);

        CutSheet.CutSheet cutSheet = await AlumCloudPlans.Manager.GetCutSheetAsync(elev);

        ...
   }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM