asp.net web api 文件上传

首先分别介绍正确的做法和错误的做法,然后分析他们的不同和错误之处,以便读者在实现此功能时可避开误区

1正确的做法

public class AvaterController : BaseApiController
{
[HttpPost]
        public async Task<IHttpActionResult> UploadAvater(int userId)
        {
            AvatarBLL pictureBLL = new AvatarBLL(this.Request);
            await pictureBLL.UploadAvatar(userId);
            return Ok();
     }

//其他Action
}

public class AvatarBLL
{
private HttpRequestMessage HttpRequestMessage;
public AvatarBLL(HttpRequestMessage httpRequestMessage)
{
     this.HttpRequestMessage = httpRequestMessage;
}
public async Task UploadAvatar(int userId)
        {
            if (!HttpRequestMessage.Content.IsMimeMultipartContent("form-data"))
            {
                //抛异常
            }
//获得客户端传递到服务器的数据
            List<byte> list = new List<byte>();
            await HttpRequestMessage.Content.ReadAsMultipartAsync().ContinueWith(multipartContent => 
            {
                if (multipartContent.IsFaulted || multipartContent.IsCanceled)
                {
                    //抛异常
                }

                foreach (var content in multipartContent.Result.Contents)
                {
                    var b = content.ReadAsByteArrayAsync().Result;
                    list.AddRange(b);
                }
         });

//其他部分(将数据存入Mongodb以及其他的业务逻辑)
}


}

2错误的做法

public class AvaterController : BaseApiController
{
[HttpPost]
        public IHttpActionResult UploadAvater(int userId)
        {
            AvatarBLL pictureBLL = new AvatarBLL(this.Request);
            pictureBLL.UploadAvatar(userId);
            return Ok();
     }

//其他Action
}
public class AvatarBLL
{
private HttpRequestMessage HttpRequestMessage;
public AvatarBLL(HttpRequestMessage httpRequestMessage)
{
     this.HttpRequestMessage = httpRequestMessage;
}
public void UploadAvatar(int userId)
        {
            if (!HttpRequestMessage.Content.IsMimeMultipartContent("form-data"))
            {
                //抛异常
            }
//获得客户端传递到服务器的数据
            List<byte> list = new List<byte>();
            MemoryStream ms = new MemoryStream();
            try
            {
                 MultipartMemoryStreamProvider mmsp = new MultipartMemoryStreamProvider();
                 var task =                          HttpRequestMessage.Content.ReadAsMultipartAsync<MultipartMemoryStreamProvider>(mmsp, 100000);
                 task.Wait();
                 var contents = task.Result.Contents;

                 foreach (var c in contents)
                 {
                    var b = c.ReadAsByteArrayAsync();
                    b.Wait();
                    list.AddRange(b.Result);
                 }
              }
              catch (AggregateException ex)
                 { }

//其他部分(将数据存入Mongodb以及其他的业务逻辑)
}
}

3 错误现象:

采用第二种方式,如果客户端上传到服务的数据量(调用UploadAvater上传的数据)小于服务端设置的缓冲区的大小,那么可正常上传文件,如果大于服务端设置的缓冲区的大小,则无法正常上传,调试服务端代码,当执行到task.Wait();这行语句时,客户端一直等待,直到客户端调用超时,永远也无法返回调用结果,发生了死锁!!!使用HttpRequestMessage.Content.ReadAsMultipartAsync<MultipartMemoryStreamProvider>(mmsp, 1000);设置缓冲区大小为1000bit。这个方法有几个重载的方法,其中一个是不显示设置缓冲区大小,那么缓冲区大小为默认的。

4 对第二种方法的错误点分析:

看Web api dll源码中的设置:

可以看出 默认的缓冲区区大小为32*1024,即32K,那么上传超过32k而不设置缓冲区大小的情况下,为什么会发生死锁,而将缓冲区设置超过上传文件大小为什么不会发生死锁呢?不论是否将缓冲区大小设置的足够大,都有发生死锁的可能。

主要的方法见上图,在方法体中有下面这段代码:

这段代码的核心方法:

上面的方法,循环读取请求数据,当设置的缓冲区大小小于客户端发送到服务器的数据量时,要执行多次循环读取数据,每次循环读取数据都是调用两个异步方法:

然而,ReadAsMultipartAsync方法的返回值是Task<T>(T为 streamProvider),所以当调用Task.Wait()方法等待的时候,ReadAsMultipartAsync方法内部也在等待异步处理streamProvider返回结果,这样就造成了死锁。

5 第一种方法为什么不会出现死锁?

第一种方法使用await,实现同步机制,而没有调用Task.Wait()方法,这样就避免了A、B两块代码块互相等待返回结果而导致死锁的可能。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java技术分享

Redis实现信息已读未读状态提示

前提: 假如现在有2个模块需要提示消息:只要存在用户在上个时间点之后没有看过的信息就提示用户有新的信息 思路如下: 使用hash存储用户上次看过的时间,使用so...

8296
来自专栏Google Dart

Dart 服务端开发 shelf_bind 包

shelf_bind倾向于约定优于配置,因此您可以编写必要的最小代码,但仍然可以根据需要覆盖默认值。

972
来自专栏magicsoar

初识nginx——内存池篇

初识nginx——内存池篇      为了自身使用的方便,Nginx封装了很多有用的数据结构,比如ngx_str_t ,ngx_array_t, ngx_poo...

24210
来自专栏安恒网络空间安全讲武堂

二进制学习系列-栈溢出之Passcode详解

概念:每一个外部定义的符号在全局偏移表(Global offset Table)中有相应的条目,GOT位于ELF的数据段中,叫做GOT段。

1604
来自专栏我的博客

PHP5.3~PHP5.5新特性汇总

一.PHP 5.3中的新特性 1. 支持命名空间 (Namespace) 2. 支持延迟静态绑定(Late Static Binding) 3. 支持got...

3908
来自专栏向治洪

命令模式

命令模式定义 将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作。 作用 命令模式主要应用于将行...

2019
来自专栏码神联盟

珍藏 | Java 岗位 100道 面试题及答案详解

9495
来自专栏Java技术分享

Redis实现信息已读未读状态提示

假如现在有2个模块需要提示消息:只要存在用户在上个时间点之后没有看过的信息就提示用户有新的信息

53110
来自专栏技术博文

phpcms v9 常用函数

常用函数 , 打开include/global.func.php,下面存放一些公共函数 view plaincopy to clipboardprint? fu...

3847
来自专栏大内老A

在ASP.NET Core应用中如何设置和获取与执行环境相关的信息?

HostingEnvironment是承载应用当前执行环境的描述,它是对所有实现了IHostingEnvironment接口的所有类型以及对应对象的统称。如下面...

7688

扫码关注云+社区

领取腾讯云代金券