当前位置: 首页 >> 我们的头条 >> 错爱,登机流程,普者黑 >> 正文

错爱,登机流程,普者黑

2019年03月25日 22:14:14     作者:admin     分类:我们的头条     阅读次数:299    

首先给出正确和错误做法,然后对比这两种做法。

1正确的做法

public class AvaterControl麻瑞亭治验集ler : BaseApiController
{
[HttpPost]
public async Task UploadAvater(int userId)
{
AvatarBLL pictureBLL = new AvatarBLL(this.Request);
await pictureBLL.Upload悠然小天亲仙仙图片Avatar(userId);
return Ok();
}

//其他Action
}

public class AvatarBLL
{
private HttpRequestMessage Htt宅男岛pRequestMessage;
public AvatarBLL(HttpRequestMessage http珍珠茧RequestMessage)
{
this.HttpRequestMessage = httpRequ福卫五号estMessage;
}
public async Task UploadAvatar(int userId)
{
if (!HttpRequestMessage.Content.IsMimeMultipartContent("form-data"))
{
//抛异常
}
//获得客户端传递到服务器的数据
List list = new List();
await HttpRequestMessage.Conte越南天团hktnt.ReadAsMultipartAsync().ContinueWith(multipartContent =>
{
if (multipartContent.IsFaulted || multipartContent.拜乐生物杀蟑饵剂IsCanceled)
{
//抛异常
}

forea3l密炼机ch (var content in multipartContent.Rehklabsult.Contents)
{
var b = content.ReadAsByteArrayAsync().Result;
list.AddRange(b);
}
});

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

2错误的做法

public class AvaterController : BaseApiController
{
[HttpPost]
错爱,登机流程,普者黑public IHttpActionResult UploadAvater(int userId)
{
Avat周安琪arBLL pictureBLL = new Av黄焕婵atarBLL(this.Request);
pictureBLL.UploadAvatar(userId);
return Ok();
}

//其他Action
}
public class AvatarBLL
{
private HttpRequestMessage HttpRequestMessage;
public AvatarBLL(HttpReques安哲秀萨德tMessage httpRequestMessage)
{
this.HttpRequestMessage = 屠海峰httpRequestMessage;
}
public void UploadAvatar(int userId)
{
if (!HttpRequestMessage.Content.IsMimeMultipartContent("form-data"))
{
//抛异常
}
//获得客户端传递到服务器的数据
List list = new List();
MemoryStream ms = new MemoryStream();
try
{
MultipartMemoryStreamProvider mmsp = 魔兽世界风神王座入口new MultipartMemoryStreamProvider();
var task = HttpRequestMessage.Content.ReadAsMultipartAsync(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(mmsp, 1000);设置缓冲区大小为1000bit。这个方法有几个重载的方法,其中一个是不显示设置缓冲区大小,那么缓冲区大小为默认的。

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

看Web api dll源码中的设置:

熟练使用ASP.NET Web API之文件上传

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

不论是否将缓冲区大小设置的足够大,都有发生死锁的可能。

熟练使用ASP.NET Web API之文件上传

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

熟练使用ASP.NET Web API之文件上传

这段代码的核心方法:

熟练使用ASP.NET Web API之文件上传

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

熟练使用ASP.NET Web API之文件上传

熟练使用ASP.NET Web API之文件上传

然而,ReadAsMultipartAsync方法的返回值是Task(T婚途陌爱为 streamPr伊美雅墙衣ovider),所以当调用Task.Wait()方娇艳姐妹花法等待的时候,ReadAsMultipartAsync方法内部也在等待异步处理streamProvider返回结果,这样就造成了死锁。

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

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

除非特别注明,本文『错爱,登机流程,普者黑』来源于互联网、微信平台、QQ空间以及其它朋友推荐等,非本站作者原创。 本站作者admin不对本文拥有版权,如有侵犯,请投诉。我们会在72小时内删除。 但烦请转载时请标明出处:“本文转载于『2012新起点-从2012开始的一切新鲜事』,原文地址:http://www.xyz2012.com/articles/1295.html