将 FormData 转换为查询字符串的更简单方法

问题描述 投票:0回答:4

我通过

XMLHttpRequest
发送 POST 请求,并将数据输入到 HTML 表单中。不受 JavaScript 干扰的表单将提交编码为
application/x-www-form-urlencoded
的数据。

使用 XMLHttpRequest,我想通过

FormData
API 发送数据,但该 API 不起作用,因为它将数据视为编码为
multipart/form-data
的数据。因此,我需要将数据作为查询字符串写入,并正确转义到
XMLHttpRequest
的 send 方法中。

addEntryForm.addEventListener('submit', function(event) {
    // Gather form data
    var formData = new FormData(this);
    // Array to store the stringified and encoded key-value-pairs.
    var parameters = []
    for (var pair of formData.entries()) {
        parameters.push(
            encodeURIComponent(pair[0]) + '=' +
            encodeURIComponent(pair[1])
        );
    }

    var httpRequest = new XMLHttpRequest();
    httpRequest.open(form.method, form.action);

    httpRequest.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');

    httpRequest.onreadystatechange = function() {
        if (httpRequest.readyState === XMLHttpRequest.DONE) {
            if (httpRequest.status === 200) {
                console.log('Successfully submitted the request');
            } else {
                console.log('Error while submitting the request');
            }
        }
    };

    httpRequest.send(parameters.join('&'));

    // Prevent submitting the form via regular request
    event.preventDefault();
});

现在整个事情与

for ... of
循环等似乎有点令人费解。有没有更简单的方法将
FormData
转换为查询字符串?或者我可以以某种方式使用不同的编码发送 FormData 吗?

javascript xmlhttprequest form-data
4个回答
130
投票

您可以使用 URLSearchParams

const queryString = new URLSearchParams(new FormData(myForm)).toString()

11
投票

您要求更简单的解决方案...

for
循环是遍历集合的最简单方法 - 恕我直言。

但是如果您使用 扩展运算符/语法 (

...
)

,则有一个更短的版本

扩展语法允许表达式在以下位置展开: 多个参数(对于函数调用)或多个元素(对于 数组文字)或多个变量(用于解构赋值) 预计。

您的

for...of
循环可以替换为:

let parameters = [...formData.entries()] // expand the elements from the .entries() iterator into an actual array
                     .map(e => encodeURIComponent(e[0]) + "=" + encodeURIComponent(e[1]))  // transform the elements into encoded key-value-pairs

5
投票

如果您可以使用

JQuery
,您只需在
.serialize()
对象上调用
form
函数,如下所示:

function getQueryString() {
  var str = $( "form" ).serialize();
  console.log(str);
}

$( "#cmdTest" ).on( "click", getQueryString );
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>

<form>
  <select name="list1">
    <option>opt1</option>
    <option>opt2</option>
  </select>
 
  <br>
  <input type="text" name="txt1" value="valWith%Special&Char$">
  <br>
  <input type="checkbox" name="check" value="check1" id="ch1">
  <label for="ch1">check1</label>
  <input type="checkbox" name="check" value="check2" checked="checked" id="ch2">
  <label for="ch2">check2</label>
 
  <br>
  <input type="radio" name="radio" value="radio1" checked="checked" id="r1">
  <label for="r1">radio1</label>
  <input type="radio" name="radio" value="radio2" id="r2">
  <label for="r2">radio2</label>
</form>
  
<button id="cmdTest">Get queryString</button>

注意:它还对数据进行编码以便在 url 请求中使用

希望对你有帮助,再见。


0
投票

打字稿解决方法:

const formData = new FormData();
const searchParams = new URLSearchParams(
  formData as unknown as Record<string, string>,
).toString();
© www.soinside.com 2019 - 2024. All rights reserved.