谈谈form-data请求格式

引言

最近在参与一个项目过程中遇到一个问题,相信大部分人都遇到过:

在后端与前端约定好application/json格式传递数据时,因为后台是go强类型语言,在定义api接口时,某些字段要求是整型类型,但是对于前端来说输入框或者从url中的search取到的参数都是字符串,不得不进行前端类型转换。

咋一看,对于接口参数比较少的api前端转换没有什么,但是对于一般的交互复杂,参数比较多的接口,要对大部分参数进行类型转换就是一种吃力不讨好的活。好在后端同学还支持另一种的前后端数据交互格式,即multipart/form-data。通过该格式后端取到前端传递的数据就是数字了(即使前端传递的是字符串),而不像json格式获取的是字符串。这样,就不需要额外对前端获取的数据进行特殊转换了。下面就来说说form-data。

form-data请求格式

multipart/form-data是基于post方法来传递数据的,并且其请求内容格式为Content-Type: multipart/form-data,用来指定请求内容的数据编码格式。另外,该格式会生成一个boundary字符串来分割请求头与请求体的,具体的是以一个boundary=${boundary}来进行分割,伪码如下:

...
Content-Type: multipart/form-data; boundary=${boundary} 

--${boundary}
...
...

--${boundary}--

上面boundary=${boundary}之后就是请求体内容了,请求体内容各字段之间以--${boundary}来进行分割,以--${boundary}--来结束请求体内容。具体可以参考下面例子:

POST http://www.example.com HTTP/1.1
Content-Type:multipart/form-data; boundary=----WebKitFormBoundaryyb1zYhTI38xpQxBK

------WebKitFormBoundaryyb1zYhTI38xpQxBK
Content-Disposition: form-data; name="city_id"

1

------WebKitFormBoundaryyb1zYhTI38xpQxBK
Content-Disposition: form-data; name="company_id"

2
------WebKitFormBoundaryyb1zYhTI38xpQxBK
Content-Disposition: form-data; name="file"; filename="chrome.png"
Content-Type: image/png

PNG ... content of chrome.png ...
------WebKitFormBoundaryyb1zYhTI38xpQxBK--

form-data格式一般是用来进行文件上传的。使用表单上传文件时,必须让

表单的 enctype 等于  multipart/form-data,因为该值默认值为 application/x-www-form-urlencoded

FormData对象

XMLHttpRequest Level 2添加了一个新的接口FormData。利用FormData对象,我们可以通过JavaScript用一些键值对来模拟一系列表单控件,我们还可以使用XMLHttpRequest的send()方法来异步的提交这个"表单"。

var formData = new FormData();
formData.append("username", "Groucho");
formData.append("accountnum", 123456); 
fetch('/users', {
  method: 'POST',
  body: formData
})

上面创建了一个FormData对象,通过fetch进行ajax请求时,会自动为其将其转为form-data格式,无需手动添加格式。

对象转FormData对象

对于FormDat对象,像上面那种形式可以直接添加参数比较方便,但是对于对象或者嵌套对象:

 let userObj = {userName: ’xxx', age: '21'}
 formData.append('user', userObj)

上面形式添加formData参数user,并不会获取到其真正的内容,而是返回userObj的Object.prototype.toString.call(userObj)的值作为user字段的值。

------WebKitFormBoundaryyb1zYhTI38xpQxBK
Content-Disposition: form-data; name="user"

[object Object]

遗憾的是,FormData对象没有像JSON.stringify那样的方法能批量将对象形式转换为对应的形式,formData而言是将对象的key转换为正确formData请求参数字段名,例如如下对象:

var obj = {
    a: '2', 
    b: {c: 'test'}, 
    c: [ 
        {id: 1, name: 'xx'}, 
        {id:2 ,name: 'yy', info: {d: 4} }
    ]
}

这样转换为FormData对象时,其对应的key应该是下面这样的:

a: 2
b[c]: test
c[][id]: 1
c[][name]: xx
c[][id]: 2
c[][name]: yy
c[][info][d]:4

这样,就需要我们自己手动来实现一个转换数据函数,具体代码如下:

function objectToFormData (obj, form, namespace) {
  const fd = form || new FormData();
  let formKey;
  
  for(var property in obj) {
      if(obj.hasOwnProperty(property)) {
        let key = Array.isArray(obj) ? '[]' : `[${property}]`;
        if(namespace) {
          formKey = namespace + key;
        } else {
          formKey = property;
        }
      
        // if the property is an object, but not a File, use recursivity.
        if(typeof obj[property] === 'object' && !(obj[property] instanceof File)) {
          objectToFormData(obj[property], fd, formKey);
        } else {
          
          // if it's a string or a File object
          fd.append(formKey, obj[property]);
        }
        
      }
    }
  
  return fd;
    
}

这样,就可以将对象转化为对应的formData的格式了。


转载地址:https://www.cnblogs.com/wonyun/p/7966967.html

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: Java可以使用Java HttpURLConnection 或 Apache Httpclient实现form-data请求。 其中HttpURLConnection是Java内置UrlConnection类的实现,通过读取form-data表单中的键值对数据,使用setRequestProperty方法设置HTTP请求头信息,然后将multipart表单数据写入输出流中,最后获取响应。 而Apache Httpclient是第三方库,也是常用的实现方式。它提供了一个MultipartEntityBuilder类,可以方便地构建form-data数据。将数据塞入MultipartEntityBuilder实例中,然后将实例传入HttpPost请求体中,最后获取响应。 无论何种实现方式,在处理form-data请求时,可以通过ContentType指定multipart/form-data的MIME类型,并且还需要指定boundary分隔符。这就是form-data请求的基本实现过程。 ### 回答2: form-data是一种数据传输格式,通常用于上传文件或提交表单数据。Java中可以使用HttpURLConnection类或HttpClient库发送form-data请求。 使用HttpURLConnection类发送form-data请求,需要先创建一个HttpURLConnection对象,并设置请求方法为POST。然后指定Content-Type为multipart/form-data,并使用OutputStream传输数据。文件数据需要使用FileInputStream读取,并按照form-data格式进行编码。 使用HttpClient库发送form-data请求,需要使用MultipartEntity类创建一个包含文件和表单数据的实体。然后使用HttpPost对象指定请求的URL,设置实体,并使用HttpClient对象执行请求。 以下是一个使用HttpClient库发送包含文件和表单数据的form-data请求的示例代码: ``` HttpEntity reqEntity = MultipartEntityBuilder.create() .addBinaryBody("file", new File("/path/to/file"), ContentType.DEFAULT_BINARY, "filename.txt") .addTextBody("form_field", "value") .build(); HttpPost httpRequest = new HttpPost("http://example.com/upload"); httpRequest.setEntity(reqEntity); HttpClient httpClient = HttpClientBuilder.create().build(); HttpResponse response = httpClient.execute(httpRequest); ``` 以上就是Java发送form-data请求的简单介绍。 ### 回答3: Java中的form-data请求是一种HTTP请求方法,它用于向服务器提交表单数据。在Java中,我们可以使用HttpURLConnection或Apache HttpClient这样的HTTP客户端库来发送form-data请求。 要发送form-data请求,我们需要创建一个HTTP POST请求,并将请求内容设置为表单数据。表单数据由键值对组成,每个键值对分别表示表单字段的名称和值。可以使用Java中的Map来存储这些键值对。这里需要注意的是,如果表单包含文件上传,则需要将文件内容编码为Base64格式,并作为值发送到服务器。 以下是一个使用Apache HttpClient发送form-data请求的示例代码: ``` CloseableHttpClient httpClient = HttpClients.createDefault(); HttpPost httpPost = new HttpPost("http://example.com/form-data-endpoint"); MultipartEntityBuilder builder = MultipartEntityBuilder.create(); builder.setMode(HttpMultipartMode.BROWSER_COMPATIBLE); //添加文本字段 builder.addTextBody("username", "John Smith", ContentType.TEXT_PLAIN); builder.addTextBody("password", "secret", ContentType.TEXT_PLAIN); //添加文件 File file = new File("/path/to/file"); builder.addBinaryBody("file", file, ContentType.APPLICATION_OCTET_STREAM, file.getName()); //设置请求实体 HttpEntity entity = builder.build(); httpPost.setEntity(entity); //发送请求 CloseableHttpResponse response = null; try { response = httpClient.execute(httpPost); //处理响应 } catch (IOException e) { //处理异常 } finally { if (response != null) { try { response.close(); } catch (IOException e) { //处理异常 } } } ``` 在上面的示例中,我们首先创建一个HttpPost对象,并指定目标URL。然后,我们使用MultipartEntityBuilder类创建一个表单实体。我们使用addTextBody方法添加文本字段,使用addBinaryBody方法添加文件。最后,我们将表单实体设置为HTTP POST请求的实体,并发送请求。如果需要,我们可以从响应中获取服务器的响应内容并进行处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值