前言:
本次要讲的是使用.Net HttpClient拼接multipark/form-data形式post上传文件和相关参数,并接收到上传文件成功后返回过来的结果(图片地址,和是否成功)。可能有很多人会说用ajax不是就可以轻松的实现吗?的确是在不存在跨域问题的前提下使用ajax上传文件,接收返回结果是最佳的选择。无奈的是我们对接的是第三方的一个上传图片的接口,而且对方并没有对我们的域名设置允许跨域,为了能够解决这一问题我们只能够通过后端请求避免跨域问题。.
什么是multipart/form-data请求:
关于multipart/form-data详情查看:https://www.cnblogs.com/tylerdonet/p/5722858.html
Html上传图片按钮:
<div class="cover-hd"><a href="javascript:;" class="a-uploadCustom"><input type="file" id="Logoimg" onchange="OnchangeImage(this)" /></a></div>
使用ajax将图片文件流和相关参数传递到后端进行拼接:
注意:因为我这里调用第三方接口需要传递(appid应用程序唯一标识,random随机数,和sign签名)
<script type="text/javascript">//后端图片上传function OnchangeImage(obj) {var formData = new FormData();var files = $(obj).prop('files'); //获取到文件列表console.log(files[0]);formData.append("imgType", 1);formData.append("appId","你需要传递的参数");formData.append("random", "你需要传递的参数");formData.append("file", files[0]);//图片文件流formData.append("sign", "你需要传递的参数");console.log(formData);jQuery.support.cors = true;$.ajax({async: true,contentType: false, //头部请求内容格式dataType: 'json',type: 'post',data:formData,// 告诉jQuery不要去处理发送的数据processData: false,url: "@Url.Action("ImageUpload", "MtVirtualStore")",//后端接收图片接口success: function(data) {//后端Httpclient请求成功后返回过来的结果console.log(data);}});}</script>
后端接收图片和参数,并将图片文件流转化为图片字节类型数据:
//接收前端图片文件信息[HttpPost]public JsonResult ImageUpload(FormContext context){HttpPostedFileBase fileData = Request.Files[0];string appId=Request["appId"];string random=Request["random"];string sign=Request["sign"];string imgType=Request["imgType"];if (fileData != null){try{string fileName = Path.GetFileName(fileData.FileName);//原始文件名称byte[] byteFileData = ReadFileBytes(fileData);//文件流转为字节流var resultContext =HttpClientPostUpload(byteFileData,appId,random,sign,imgType, fileName);return Json(new { code = 1, list = resultContext,msg="上传成功~"});}catch (Exception ex){return Json(new { code = 0, msg = ex.Message });}}else{return Json(new { code = 0, msg = "图片上传失败,请稍后再试~" });}}//文件流转化为字节/// <summary>/// 文件流类型转化字节类型/// </summary>/// <param name="fileData">文件流数据</param>/// <returns></returns>private byte[] ReadFileBytes(HttpPostedFileBase fileData){byte[] data;using (Stream inputStream = fileData.InputStream){MemoryStream memoryStream = inputStream as MemoryStream;if (memoryStream == null){memoryStream = new MemoryStream();inputStream.CopyTo(memoryStream);}data = memoryStream.ToArray();}return data;}
重点,HttpClient拼接multipart/form-data形式参数post提交数据:
/// <summary>/// 向目标地址提交图片文件参数数据/// </summary>/// <param name="bmpBytes">图片字节流</param>/// <param name="appId">appid</param>/// <param name="random">随机数</param>/// <param name="sign">签名</param>/// <param name="imgType">上传图片类型</param>/// <param name="fileName">图片名称</param>/// <returns></returns>public string HttpClientPostUpload(byte [] bmpBytes, string appId, string random,string sign,string imgType,string fileName){using (var client = new HttpClient()){List<ByteArrayContent> list = new List<ByteArrayContent>();var dataContent = new ByteArrayContent(Encoding.UTF8.GetBytes(appId));dataContent.Headers.ContentDisposition = new ContentDispositionHeaderValue("form-data")//内容处置标头{Name = "appId"};list.Add(dataContent);var dataContent2 = new ByteArrayContent(Encoding.UTF8.GetBytes(imgType));dataContent2.Headers.ContentDisposition = new ContentDispositionHeaderValue("form-data"){Name = "imgType"};list.Add(dataContent2);var dataContent3 = new ByteArrayContent(Encoding.UTF8.GetBytes(random));dataContent3.Headers.ContentDisposition = new ContentDispositionHeaderValue("form-data"){Name = "random"};list.Add(dataContent3);var dataContent4 = new ByteArrayContent(Encoding.UTF8.GetBytes(sign));dataContent4.Headers.ContentDisposition = new ContentDispositionHeaderValue("form-data"){Name = "sign"};list.Add(dataContent4);var fileContent = new ByteArrayContent(bmpBytes);//填充图片字节fileContent.Headers.ContentDisposition = new ContentDispositionHeaderValue("form-data"){Name="file",FileName=fileName};list.Add(fileContent);using (var content =new MultipartFormDataContent()){Action<List<ByteArrayContent>> act = (dataContents) =>{//声明一个委托,该委托的作用就是将ByteArrayContent集合加入到MultipartFormDataContent中foreach (var byteArrayContent in dataContents){content.Add(byteArrayContent);}};act(list);//执行acttry{var result = client.PostAsync("https://xxxxxx.com/imageUpload/", content).Result;//post请求return result.Content.ReadAsStringAsync().Result;}catch (Exception ex){return ex.Message;}}}}
使用Fiddler 4 抓包查看请求的参数:
因为我们没有办法看到我们所拼接成功后的multipark/form-data形式的数据,想要看到对应拼接的请求参数可以使用 Fiddler 4 抓包工具查看:
关于Fiddler 4抓包工具的使用可以阅读该篇博客:https://www.jianshu.com/p/55f7be58a7e4
抓包获取到的multipark/form-data形式的请求参数如下图:

