在HTTP服务应用中进行数据提交一般都使用application/json,application/x-www-form-urlencodedmultipart/form-data这几种内容格式。这几种格式的处理复杂度处理起来和前面定义的先后顺序一样由易到难。不过现有工具都提供了完善的功能在提交这些数据的时候都比较方便了;不过要自己手动基础协议写起,那multipart/form-data的处理规范还是要相对复杂些。最近在写webapi管理和性能测试工具(https://github.com/IKende/WebBenchmark)时为了得到更可控的时间线和性能,在实现并没有用到任何应用组件都是从HTTP基础协议写起,在这时介绍一下如何在基础HTTP协议的基础上提交multipart/form-data数据.(如果你没有什么特别的需求还是不要这么干)

multipart/form-data

这种格式一般配合多数据类型提交使用,如常用的数据表单和文件结合。这种格式有着自己的处理规范和application/jsonapplication/x-www-form-urlencoded有着不同。application/json相对来说最简单整个数据流是json内容格式,而application/x-www-form-urlencoded则是以k-v的方式处理,只是对应的值要做Url编写。而multipart/form-data则用一个特别的分隔符来处理,这个分隔符分为开始分隔和结束分隔符。

分隔符定义

如果使用multipart/form-data提交数据,那必须在Content-Type的请求头后面添加; boundary=value这样一个描述,boundary的值即是每项数据之间的分隔符

  1. mHeaderCached.Append("Content-Type: ").Append(mCases.ContentType);
  2. if (multipartFormData)
  3. mHeaderCached.Append("; boundary=").Append(boundary);
  4. mHeaderCached.Append("\r\n");

需要怎样定义boundary值?其实boundary的定义是没有特别的要求的,就是一个字符串完全看自己的喜好。但最终处理的时候是要有一个规范。

  • 开始分隔符--boundary

  • 结束分隔符--boundary--

开始分隔符必须在每项数据之前写入,简单来说就是有多少项数据就有多少个开始分隔符了;结束分隔符只有一个,就是在提交内容的尾部添加,说明这个提交的内容在这里结束不需要再往下解释。大概格式如下:

  1. -- boundary
  2. 数据项
  3. -- boundary
  4. 数据项
  5. -- boundary
  6. 数据项
  7. -- boundary
  8. 数据项
  9. --boundary--

数据项

multipart/form-data中的每项数据都分别有HeaderBody和整个HTTP上层协议差不多。

  1. Content-Disposition: form-data; name="fname"\r\n
  2. \r\n
  3. value
  4. \r\n

Content-Disposition是必须的,描述一下这数据的格式来源,在这里都是form-data;后面根据不同数据的情况有着不同的属性,每个属性用;分隔的K-V结构。代码的处理比较简单:

  1. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"");

接下来就是一个空换行然后再写入值,完整代码如下:

  1. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"");
  2. mMemoryData.WriteLine("");
  3. mTextBodyCached.Clear();
  4. item.GetTemplate().Execute(mTextBodyCached);
  5. mMemoryData.Write(mTextBodyCached);
  6. mMemoryData.WriteLine("");

提交文件

提交文件相对来说比值要处理多一些属性,主要包括内容类型,文件名等;其实写起来也不复杂

  1. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"; filename=\"{item.FileName}\"");
  2. mMemoryData.WriteLine($"Content-Type: {item.Type}");
  3. mMemoryData.WriteLine("");
  4. var itemBuffer = item.GetBuffer();
  5. mMemoryData.Write(itemBuffer, , itemBuffer.Length);
  6. mMemoryData.WriteLine("");

以上就是multipart/form-data普通值和文件提交时写的数据格式,下面看一下这个multipart/form-data的完整代码

  1. for (int i = ; i < mCases.FormBody.Count; i++)
  2. {
  3. var item = mCases.FormBody[i];
  4. mMemoryData.Write("--");
  5. mMemoryData.WriteLine(boundary);
  6. if (item.Type == HttpDataType.Bytes)
  7. {
  8. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"; filename=\"{item.FileName}\"");
  9. mMemoryData.WriteLine($"Content-Type: {item.Type}");
  10. mMemoryData.WriteLine("");
  11. var itemBuffer = item.GetBuffer();
  12. mMemoryData.Write(itemBuffer, , itemBuffer.Length);
  13. mMemoryData.WriteLine("");
  14. }
  15. else
  16. {
  17. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"");
  18. mMemoryData.WriteLine("");
  19. mTextBodyCached.Clear();
  20. item.GetTemplate().Execute(mTextBodyCached);
  21. mMemoryData.Write(mTextBodyCached);
  22. mMemoryData.WriteLine("");
  23. }
  24. }
  25. if (mCases.FormBody.Count > )
  26. {
  27. mMemoryData.Write("--");
  28. mMemoryData.Write(boundary);
  29. mMemoryData.WriteLine("--");
  30. mMemoryData.Flush();
  31. }

这样一个完整的multipart/form-data提交基础协议代码就处理完成;在webbenchmark的实现有还有application/jsonapplication/x-www-form-urlencoded的处理,相对于multipart/form-data来说这两个处理就更加简单了;下面包括:POST,GET,PUT,DELETE和三种数据格式提交的完整代码函(在BeetleX的pipestream帮助下这些协议的处理还是比较简单的)

  1. public void Write(PipeStream stream)
  2. {
  3. string boundary = null;
  4. bool multipartFormData = mCases.ContentType == "multipart/form-data";
  5. if (multipartFormData)
  6. boundary = "----Beetlex.io" + DateTime.Now.ToString("yyyyMMddHHmmss");
  7. byte[] bodyData = null;
  8. int bodyLength = 0;
  9. if (mHeaderCached == null)
  10. mHeaderCached = new StringBuilder();
  11. mHeaderCached.Clear();
  12.  
  13. if (mMemoryData == null)
  14. mMemoryData = new PipeStream();
  15. if (mMemoryData.Length > 0)
  16. mMemoryData.ReadFree((int)mMemoryData.Length);
  17.  
  18. if (mTextBodyCached == null)
  19. mTextBodyCached = new StringBuilder();
  20. mTextBodyCached.Clear();
  21.  
  22. mHeaderCached.Append(mCases.Method).Append(" ");
  23. mUrlTemplate.Execute(mHeaderCached);
  24. for (int i = 0; i < mCases.QueryString.Count; i++)
  25. {
  26. if (i == 0)
  27. {
  28. if (mUrlHasParameter)
  29. mHeaderCached.Append("&");
  30. else
  31. mHeaderCached.Append("?");
  32. }
  33. else
  34. {
  35. mHeaderCached.Append("&");
  36. }
  37. mHeaderCached.Append(mCases.QueryString[i].Name);
  38. mHeaderCached.Append("=");
  39. mCases.QueryString[i].GetTemplate().Execute(mHeaderCached, true);
  40. }
  41. mHeaderCached.Append(" ");
  42. mHeaderCached.Append(Protocol).Append("\r\n");
  43.  
  44. foreach (var item in mCases.Header)
  45. {
  46. mHeaderCached.Append(item.Name).Append(": ");
  47. item.GetTemplate().Execute(mHeaderCached);
  48. mHeaderCached.Append("\r\n");
  49. }
  50. mHeaderCached.Append("Content-Type: ").Append(mCases.ContentType);
  51. if (multipartFormData)
  52. mHeaderCached.Append("; boundary=").Append(boundary);
  53. mHeaderCached.Append("\r\n");
  54. if (multipartFormData)
  55. {
  56. for (int i = 0; i < mCases.FormBody.Count; i++)
  57. {
  58. var item = mCases.FormBody[i];
  59. mMemoryData.Write("--");
  60. mMemoryData.WriteLine(boundary);
  61. if (item.Type == HttpDataType.Bytes)
  62. {
  63. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"; filename=\"{item.FileName}\"");
  64. mMemoryData.WriteLine($"Content-Type: {item.Type}");
  65. mMemoryData.WriteLine("");
  66. var itemBuffer = item.GetBuffer();
  67. mMemoryData.Write(itemBuffer, 0, itemBuffer.Length);
  68. mMemoryData.WriteLine("");
  69. }
  70. else
  71. {
  72. mMemoryData.WriteLine($"Content-Disposition: form-data; name=\"{item.Name}\"");
  73. mMemoryData.WriteLine("");
  74. mTextBodyCached.Clear();
  75. item.GetTemplate().Execute(mTextBodyCached);
  76. mMemoryData.Write(mTextBodyCached);
  77. mMemoryData.WriteLine("");
  78. }
  79. }
  80. if (mCases.FormBody.Count > 0)
  81. {
  82. mMemoryData.Write("--");
  83. mMemoryData.Write(boundary);
  84. mMemoryData.WriteLine("--");
  85. mMemoryData.Flush();
  86. }
  87.  
  88. }
  89. else if (mCases.ContentType == "application/json")
  90. {
  91. if (mJsonBodyTemplate != null)
  92. {
  93. mJsonBodyTemplate.Execute(mTextBodyCached);
  94. }
  95. }
  96. else
  97. {
  98. for (int i = 0; i < mCases.FormBody.Count; i++)
  99. {
  100. if (i > 0)
  101. {
  102. mTextBodyCached.Append("&");
  103. }
  104. mTextBodyCached.Append(mCases.FormBody[i].Name).Append("=");
  105. mCases.FormBody[i].GetTemplate().Execute(mTextBodyCached, true);
  106. }
  107. }
  108. try
  109. {
  110. if (multipartFormData)
  111. {
  112. bodyLength = (int)mMemoryData.Length;
  113. if (bodyLength > 0)
  114. {
  115. bodyData = System.Buffers.ArrayPool<byte>.Shared.Rent(bodyLength);
  116. mMemoryData.Read(bodyData, 0, bodyLength);
  117. }
  118. }
  119. else
  120. {
  121. if (mTextBodyCached.Length > 0)
  122. {
  123. char[] charbuffer = System.Buffers.ArrayPool<char>.Shared.Rent(mTextBodyCached.Length);
  124. try
  125. {
  126. mTextBodyCached.CopyTo(0, charbuffer, 0, mTextBodyCached.Length);
  127. bodyData = System.Buffers.ArrayPool<byte>.Shared.Rent(mTextBodyCached.Length * 6);
  128. bodyLength = Encoding.UTF8.GetBytes(charbuffer, 0, mTextBodyCached.Length, bodyData, 0);
  129. }
  130. finally
  131. {
  132. System.Buffers.ArrayPool<char>.Shared.Return(charbuffer);
  133. }
  134. }
  135. }
  136. mHeaderCached.Append("Content-Length: ").Append(bodyLength).Append("\r\n");
  137. mHeaderCached.Append("\r\n");
  138. stream.Write(mHeaderCached);
  139. if (bodyData != null)
  140. {
  141. stream.Write(bodyData, 0, bodyLength);
  142. }
  143. }
  144. finally
  145. {
  146. if (bodyData != null)
  147. System.Buffers.ArrayPool<byte>.Shared.Return(bodyData);
  148. }
  149.  
  150. }

  

  1.  

从零开始实现multipart/form-data数据提交的更多相关文章

  1. Sending forms through JavaScript[form提交 form data]

    https://developer.mozilla.org/en-US/docs/Learn/HTML/Forms/Sending_forms_through_JavaScript As in the ...

  2. html5 file upload and form data by ajax

    html5 file upload and form data by ajax 最近接了一个小活,在短时间内实现一个活动报名页面,其中遇到了文件上传. 我预期的效果是一次ajax post请求,然后在 ...

  3. Form表单提交数据的几种方式

    一.submit提交 在form标签中添加Action(提交的地址)和method(post),且有一个submit按钮(<input type='submit'>)就可以进行数据的提交, ...

  4. 关于AJAX与form表单提交数据的格式

    一 form表单传输文件的格式: 只有三种: multipart/form-data 一般用于传输文件,图片文件或者其他的. 那么其中我们默认的是application/x-www-form-urle ...

  5. 表单提交数据格式form data

    前言: 最近遇到的最多的问题就是表单提交数据格式问题了. 常见的三种表单提交数据格式,分别举例说明:(项目是vue的框架) 1.application/x-www-form-urlencoded 提交 ...

  6. 使用form表单提交请求如何获取后台返回的数据?

    问题描述 一般的form表单提交是单向的:只能给服务器发送数据,但是无法获取服务器返回的数据,也就是无法读取HTTP应答包. 想要真正的半双工通讯一般需要使用Ajax, 但是Ajax对文件传输也很麻烦 ...

  7. ligerui_实际项目_003:form中添加数据,表格(grid)里面显示,最后将表格(grid)里的数据提交到servlet

    实现效果: "Form"中填写数据,向本页"Grid"中添加数据,转换成Json数据提交,计算总和,Grid文本框可编辑,排序 图片效果: 总结: //disp ...

  8. Ajax模拟Form表单提交,含多种数据上传

    ---恢复内容开始--- Ajax提交表单.使用FormData提交表单数据和上传的文件(这里的后台使用C#获取,你可以使用Java一样获取) 有时候前台的数据提交到后台,不想使用form表单上传,希 ...

  9. springMVC中对HTTP请求form data和request payload两种数据发送块的后台接收方式

    最近在做项目中发现,前台提交数据时,如果通过form表单提交和ajax发送json时,springMVC后台接收不能都通过@ModelAttribute方式处理,经过一番查找后,ajax发送json请 ...

随机推荐

  1. 3、react-props/state

    1.react中属性props和状态state 属性--静态得,所以在初始化得时候使用得是static进行初始化得,正常情况下属性不改 状态--动态得,它得值是可以发生改变得,react中的组件更新( ...

  2. 04.Java基础语法

    一.Java源程序结构与编程规范 一个完整的Java源程序应该包含下列部分 package语句,至多一句,必须放在源程序第一句 import语句,没有或者若干句,必须放在所有类定义前 public c ...

  3. MySQL数据库离线包安装与注册

    本文主要介绍了MySQL数据库的离线安装和将MySQL服务注册为Windows应用服务的主要步骤. 1.下在安装程序包 MySQL Community Server 5.6.15 官方下载地址http ...

  4. flutter-web利用dart js 库发起http request

    初学flutter,初学前端,尝试在dart中直接使用HttpClient时,直接报出Platform not supported,查资料发现他还不支持浏览器. 通过查阅资料发现可以借助axios 与 ...

  5. swift - TextView和TextField之return隐藏回收键盘

    一.点击界面空白处即可收起键盘,空白处不能有其他控件的响应事件. //点击空白处关闭键盘 override func touchesEnded(_ touches: Set<UITouch> ...

  6. postman切换环境

    原文链接:https://www.cnblogs.com/nicole-zhang/p/11498384.html 通常会有多个测试环境,针对同一个接口来说,可能只是域名有变化,此时可以添加postm ...

  7. 浅淡i.MX8M Mini处理器的效能以及平台对比

    i.MX 8M Mini是恩智浦首款嵌入式多核应用处理器,定位在任何通用工业和物联网的应用,是一款针对边缘计算应用的芯片,也是恩智普i.MX系列中第一个加了机器学习核的产品线.这颗芯片采用先进的14L ...

  8. 网页中为什么常用AT替换@(repost from https://zhidao.baidu.com/question/122291.html)

    经常在个人主页上看到别人的邮箱地址中@被AT符号替代,很是迷惑,这样替代有什么好处呢?还是说html原有的原因使界面中不能出现@,查阅资料后解答如下: 写成AT [at],是为了防止被一些邮件扫描器搜 ...

  9. 最新 iOS 框架整体梳理(三)

    这一篇得把介绍框架这个系列终结了,不能超过三篇了,不然太长了..... 还是老规矩,前面两篇的机票在下方: 最新 iOS 框架整体梳理(一) 最新 iOS 框架整体梳理(二) Part - 3     ...

  10. 利用VS自带发布功能实现web项目快速部署

    你还在使用最原始的方法部署服务器吗?还在把项目文件全部复制然后黏贴到服务器上?这种方法太low而且又慢又不安全(存在源码泄露等安全性问题),如果你是这样,那你自己肯定也为此烦恼不已. 下面我所要讲到的 ...