细谈 Web Api 图片上传,在使用 Task.ContinueWith 变量无法赋值问题的解决办法!

在使用Asp.Net Web Api 图片上传接口的时候,到网上找了一些个例子,但大多数找到都是这个版本!

[HttpPost] 
public Task<Hashtable> ImgUpload() 
{ 
    // 检查是否是 multipart/form-data 
    if (!Request.Content.IsMimeMultipartContent("form-data")) 
        throw new HttpResponseException(HttpStatusCode.UnsupportedMediaType); 
    //文件保存目录路径 
    string SaveTempPath = "~/SayPlaces/" + "/SayPic/SayPicTemp/"; 
    String dirTempPath = HttpContext.Current.Server.MapPath(SaveTempPath); 
    // 设置上传目录 
    var provider = new MultipartFormDataStreamProvider(dirTempPath); 
    //var queryp = Request.GetQueryNameValuePairs();//获得查询字符串的键值集合 
    var task = Request.Content.ReadAsMultipartAsync(provider). 
        ContinueWith<Hashtable>(o => 
        { 
            Hashtable hash = new Hashtable(); 
            hash["error"] = 1; 
            hash["errmsg"] = "上传出错"; 
            var file = provider.FileData[0];//provider.FormData 
            string orfilename = file.Headers.ContentDisposition.FileName.TrimStart('"').TrimEnd('"'); 
            FileInfo fileinfo = new FileInfo(file.LocalFileName);                     
            //最大文件大小 
            int maxSize = 10000000; 
            if (fileinfo.Length <= 0) 
            { 
                hash["error"] = 1; 
                hash["errmsg"] = "请选择上传文件。"; 
            } 
            else if (fileinfo.Length > maxSize) 
            { 
                hash["error"] = 1; 
                hash["errmsg"] = "上传文件大小超过限制。"; 
            } 
            else
            { 
                string fileExt = orfilename.Substring(orfilename.LastIndexOf('.')); 
                //定义允许上传的文件扩展名 
                String fileTypes = "gif,jpg,jpeg,png,bmp"; 
                if (String.IsNullOrEmpty(fileExt) || Array.IndexOf(fileTypes.Split(','), fileExt.Substring(1).ToLower()) == -1) 
                { 
                    hash["error"] = 1; 
                    hash["errmsg"] = "上传文件扩展名是不允许的扩展名。"; 
                } 
                else
                { 
                    String ymd = DateTime.Now.ToString("yyyyMMdd", System.Globalization.DateTimeFormatInfo.InvariantInfo); 
                    String newFileName = DateTime.Now.ToString("yyyyMMddHHmmss_ffff", System.Globalization.DateTimeFormatInfo.InvariantInfo); 
                    fileinfo.CopyTo(Path.Combine(dirTempPath, newFileName + fileExt), true); 
                    fileinfo.Delete(); 
                    hash["error"] = 0; 
                    hash["errmsg"] = "上传成功"; 
                } 
            } 
            return hash; 
        }); 
    return task; 
}
View Code

如果只是上传,简单用是可以的,但是你可能不会发现有什么问题。但如果你在 Request.Content.ReadAsMultipartAsync(provider) .ContinueWith 延时Task任务 里面赋值一个变量,你就会发现 始终赋值不上,不信你可以试试。

例子 如下:

public string UploadFile()
{
        if (Request.Content.IsMimeMultipartContent())
        {
            //Save file
            MultipartFormDataStreamProvider provider = new MultipartFormDataStreamProvider(HttpContext.Current.Server.MapPath("~/Files"));string filename = "Not set";

            Request.Content.ReadAsMultipartAsync(provider).ContinueWith(o =>
            {
                //File name
                filename = "Set success";
            }, TaskScheduler.FromCurrentSynchronizationContext()); 

            return filename;
        }
        else
        {
            return "Invalid.";
        }
 }

上面的得出的结果 :  filename = " Not set " ; 

【 注意如下结论 】

经测试发现如下结论,在执行 Request.Content.ReadAsMultipartAsync(provider) . ContinueWith 异步延时任务的时候,先不会被立即执行。

等待 return 结束之后才会被执行。这也就是为什么返回的总是: " Not set " 

经过几天的摸索测试,在StackOverFlow上找到了一个解决的办法如下: 

IEnumerable<HttpContent> parts = null;
Task.Factory
    .StartNew(() => parts = Request.Content.ReadAsMultipartAsync().Result.Contents,
        CancellationToken.None,
        TaskCreationOptions.LongRunning, // guarantees separate thread
        TaskScheduler.Default)
    .Wait();

改造后就变成了这样,真的太棒了!

public string UploadFile()
        {
            if (Request.Content.IsMimeMultipartContent())
            {
                //Save file
                MultipartFormDataStreamProvider provider = new MultipartFormDataStreamProvider(HttpContext.Current.Server.MapPath("/UploadUser/"));

                string filename = "Not set";

                IEnumerable<HttpContent> parts = null;
                Task.Factory
                    .StartNew(() =>
                    {
                        parts = Request.Content.ReadAsMultipartAsync(provider).Result.Contents;
                        filename = "Set Success";
                    },
                    CancellationToken.None,
                    TaskCreationOptions.LongRunning, // guarantees separate thread
                    TaskScheduler.Default)
                    .Wait();

                return filename;
            }
            else
            {
                return "Invalid.";
            }
        }

相关Task的文章:

http://stackoverflow.com/questions/10502353/task-continuewith-execution-orderTa

http://www.strathweb.com/2012/08/a-guide-to-asynchronous-file-uploads-in-asp-net-web-api-rtm/

StackOverFlow 最终解决方案:

http://stackoverflow.com/questions/15201255/request-content-readasmultipartasync-never-returns

### 回答1: Task.ContinueWith 方法允许您在任务完成后执行其他代码。以下是示例代码: ``` Task task = Task.Run(() => { // 执行一些操作 }); task.ContinueWith(t => { // 在任务完成后执行的代码 Console.WriteLine("任务已完成!"); }); ``` 在这个例子中,当任务完成后,控制台将输出“任务已完成!”。 ### 回答2: task.ContinueWith是一个用于任务(Task)的扩展方法,用于指定在任务完成后要执行的操作。它接受一个参数,即要执行的继续操作(continue action)。 使用task.ContinueWith可以链式地连接多个任务,使它们按照指定的顺序依次执行。 在执行task.ContinueWith之前,我们首先需要创建一个任务。任务是一种表示异步操作的对象,可以使用Task类来创建和管理任务。 当任务完成时,它将调用指定的继续操作。继续操作可以是一个委托方法,也可以是一个Lambda表达式。 继续操作可以用来处理任务的结果,例如对任务的返回值进行处理,或者执行一些后续操作。 除了继续操作外,我们还可以通过task.ContinueWith方法指定一些任务的参数,例如任务执行的选项、任务的调度器等。 在继续操作中,我们可以使用TaskStatus枚举来判断任务的状态,根据不同的状态来执行不同的操作。 需要注意的是,在继续操作中执行的代码是在任务完成后立即执行的,不会阻塞当前线程。 通过task.ContinueWith方法,我们可以方便地组织和管理多个任务,实现复杂的异步操作,并在任务完成时执行指定的操作。 ### 回答3: Task.ContinueWith代码是一个用于在一个任务完成后继续执行特定操作的方法。它可以让我们在任务完成后执行指定的操作,例如创建一个新的任务、处理任务的结果、将任务的结果传递给其他任务等。 当我们使用异步编程时,经常会遇到需要在某个任务完成后执行其他操作的情况。Task.ContinueWith方法允许我们指定一个委托,当任务完成后,这个委托将被调用。 Task.ContinueWith方法有几个不同的重载,我们可以根据需要选择使用。其中最常用的一种是将ContinuationOptions参数传递给方法,来指定任务的执行条件和行为。 使用Task.ContinueWith方法的一个常见示例是在一个任务执行完毕后,将其结果传递给另一个任务继续处理。这对于任务之间的串行处理非常有用,可以充分利用并行编程带来的性能优势。 另一个例子是在任务执行完毕后更新UI界面。在WPF或WinForms等GUI应用程序中,我们不允许在后台线程直接访问UI元素,因此使用Task.ContinueWith可以很方便地在任务完成后切回UI线程进行UI更新操作。 总之,Task.ContinueWith方法是一个非常有用的工具,可以让我们在任务完成后执行特定操作。它提供了灵活的方式来处理任务的结果或在任务之间建立顺序执行的关系。通过合理使用Task.ContinueWith,可以更好地利用异步编程的优势,提高程序的性能和响应能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值