c# – MVC4 WebApi进程启动器

我有一个MVC4 API REST尝试在新线程中启动进程.我正在使用框架4.5并尝试使用sync和等待clausules.

我的代码看起来像:

[AcceptVerbs("POST")]
public HttpResponseMessage Launch(string id)
{
    runProcessAsync(id);   // 1                                                        

    return Request.CreateResponse(HttpStatusCode.Accepted); // 2
}

protected async void runProcessAsync(string id)
{
    int exitcode = await runProcess(id);  // 3
    string exitcodesz = string.Format("exitcode: {0}",exitcode);  // 4
    // do more stuff with exitcode
}

protected Task<int> runProcess(string id)
{
    var tcs = new TaskCompletionSource<int>();

    var process = new Process
    {
        StartInfo = { FileName = @"C:\a_very_slow_task.bat" },EnableRaisingEvents = true
        };
        process.Exited += (sender,args) => tcs.SetResult(((Process)sender).ExitCode);            

        process.Start();

        return tcs.Task;
    }  
}

理想情况下,有人会使用POST动词执行api rest调用(/ task / slowtask / launch),并期望202(接受)非常快.

使用Fiddler Web调试器我发出请求,代码进入Launch(// 1),然后进入await(// 3),创建并启动慢速任务并返回Accepted(// 2).但在这一点上,Fiddler没有显示202结果.见附图:

http://imageshack.us/photo/my-images/703/fiddler1.png/

当慢速任务结束时,代码继续捕获exitcode(// 4),然后202被捕获到Fiddler中.

这很奇怪,因为我很久以前做过回归.我错过了什么?我怎样才能更改代码以便快速返回202而忘记任务.

注意:我知道如何使用非框架4.5功能执行此操作,我正在尝试学习如何使用async / await.

解决方法

我认为这是因为runProcessAsync()在请求的同步上下文上运行.如果您不想这样,可以使用Task.Run(()=> runProcessAsync(id));.但要小心这一点,因为IIS可以在此期间回收您的AppDomain,因此runProcessAsync()有可能无法完成.

相关文章

在项目中使用SharpZipLib压缩文件夹的时候,遇到如果目录较深,则压缩包中的文件夹同样比较深的问题。比...
项目需要,几十万张照片需要计算出每个照片的特征值(调用C++编写的DLL)。 业务流程:选择照片...
var array = new byte[4]; var i = Encoding.UTF8.GetBytes(100.ToString(&quot;x2&quot;));//...
其实很简单,因为Combox的Item是一个K/V的object,那么就可以把它的items转换成IEnumerable&lt;Dic...
把.net4.6安装包打包进安装程序。 关键脚本如下: 头部引用字符串对比库 !include &quot;WordFunc....
项目需求(Winform)可以批量打印某个模板,经过百度和摸索,使用iTextSharp+ZXing.NetʿreeSp...