/
index.md
158 lines (120 loc) · 6.04 KB
/
index.md
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
---
title: FormData 对象的使用
slug: Web/API/XMLHttpRequest_API/Using_FormData_Objects
---
FormData 对象用以将数据编译成键值对,以便用[`XMLHttpRequest`](/zh-CN/docs/Web/API/XMLHttpRequest)来发送数据。其主要用于发送表单数据,但亦可用于发送带键数据 (keyed data),而独立于表单使用。如果表单`enctype`属性设为 multipart/form-data,则会使用表单的{{domxref("HTMLFormElement.submit","submit()")}}方法来发送数据,从而,发送数据具有同样形式。
## 从零开始创建 FormData 对象
你可以自己创建一个`FormData`对象,然后调用它的{{domxref("FormData.append","append()")}}方法来添加字段,像这样:
```js
var formData = new FormData();
formData.append("username", "Groucho");
formData.append("accountnum", 123456); //数字 123456 会被立即转换成字符串 "123456"
// HTML 文件类型 input,由用户选择
formData.append("userfile", fileInputElement.files[0]);
// JavaScript file-like 对象
var content = '<a id="a"><b id="b">hey!</b></a>'; // 新文件的正文
var blob = new Blob([content], { type: "text/xml" });
formData.append("webmasterfile", blob);
var request = new XMLHttpRequest();
request.open("POST", "http://foo.com/submitform.php");
request.send(formData);
```
> **备注:** 字段 "userfile" 和 "webmasterfile" 都包含一个文件。字段 "accountnum" 是数字类型,它将被[`FormData.append()`](/zh-CN/docs/Web/API/FormData/append) 方法转换成字符串(字段类型可以是 {{ domxref("Blob") }}、{{ domxref("File") }} 或者字符串:**如果它的字段类型不是 Blob 也不是 File,则会被转换成字符串类**)。
上面的示例创建了一个 `FormData` 实例,包含"username"、"accountnum"、"userfile" 和 "webmasterfile"四个字段,然后使用 `XMLHttpRequest` 的 [`send()`](/zh-CN/docs/Web/API/XMLHttpRequest/send) 方法发送表单数据。字段 "webmasterfile" 是 {{domxref("Blob")}} 类型。一个 **Blob** 对象表示一个不可变的,原始数据的类似文件对象。Blob 表示的数据不一定是一个 JavaScript 原生格式。 [`File`](/zh-CN/docs/Web/API/File) 接口基于 Blob,继承 blob 功能并将其扩展为支持用户系统上的文件。你可以通过 [`Blob()`](/zh-CN/docs/Web/API/Blob/Blob) 构造函数创建一个 Blob 对象。
## 通过 HTML 表单创建 FormData 对象
想要构造一个包含 Form 表单数据的 FormData 对象,需要在创建 FormData 对象时指定表单的元素。
> **备注:** FormData 将仅使用具有 name 属性的输入字段。
```js
var formData = new FormData(someFormElement);
```
示例:
```js
var formElement = document.querySelector("form");
var request = new XMLHttpRequest();
request.open("POST", "submitform.php");
request.send(new FormData(formElement));
```
你还可以在创建一个包含 Form 表单数据的 FormData 对象之后和发送请求之前,附加额外的数据到 FormData 对象里,像这样:
```js
var formElement = document.querySelector("form");
var formData = new FormData(formElement);
var request = new XMLHttpRequest();
request.open("POST", "submitform.php");
formData.append("serialnumber", serialNumber++);
request.send(formData);
```
这样你就可以在发送请求之前自由地附加不一定是用户编辑的字段到表单数据里。
## 使用 FormData 对象上传文件
你还可以使用 FormData 上传文件。使用的时候需要在表单中添加一个文件类型的 input:
```html
<form enctype="multipart/form-data" method="post" name="fileinfo">
<label>Your email address:</label>
<input
type="email"
autocomplete="on"
autofocus
name="userid"
placeholder="email"
required
size="32"
maxlength="64" /><br />
<label>Custom file label:</label>
<input type="text" name="filelabel" size="12" maxlength="32" /><br />
<label>File to stash:</label>
<input type="file" name="file" required />
<input type="submit" value="Stash the file!" />
</form>
<div></div>
```
然后使用下面的代码发送请求:
```js
var form = document.forms.namedItem("fileinfo");
form.addEventListener(
"submit",
function (ev) {
var oOutput = document.querySelector("div"),
oData = new FormData(form);
oData.append("CustomField", "This is some extra data");
var oReq = new XMLHttpRequest();
oReq.open("POST", "stash.php", true);
oReq.onload = function (oEvent) {
if (oReq.status == 200) {
oOutput.innerHTML = "Uploaded!";
} else {
oOutput.innerHTML =
"Error " +
oReq.status +
" occurred when trying to upload your file.<br />";
}
};
oReq.send(oData);
ev.preventDefault();
},
false,
);
```
> **备注:** 如果 FormData 对象是通过表单创建的,则表单中指定的请求方式会被应用到方法 open() 中。
你还可以直接向 FormData 对象附加 File 或 Blob 类型的文件,如下所示:
```js
data.append("myfile", myBlob, "filename.txt");
```
使用 append() 方法时,可以通过第三个可选参数设置发送请求的头 `Content-Disposition` 指定文件名。如果不指定文件名(或者不支持该参数时),将使用名字“blob”。
如果你设置正确的配置项,你也可以通过 jQuery 来使用 FormData 对象:
```js
var fd = new FormData(document.querySelector("form"));
fd.append("CustomField", "This is some extra data");
$.ajax({
url: "stash.php",
type: "POST",
data: fd,
processData: false, // 不处理数据
contentType: false, // 不设置内容类型
});
```
## 不使用 FormData 对象,通过 AJAX 提交表单和上传文件
如果你想知道不使用 FormData 对象的情况下,如何序列化并通过 [AJAX](/zh-CN/docs/AJAX) 提交表单,[请点击这里](/zh-CN/docs/Web/API/XMLHttpRequest_API/Using_XMLHttpRequest#提交表单和上传文件)。
## 参见
- [使用 XMLHttpRequest](/zh-CN/docs/Web/API/XMLHttpRequest_API/Using_XMLHttpRequest)
- {{domxref("HTMLFormElement")}}
- {{domxref("Blob")}}
- [类型化数组](/zh-CN/docs/Web/JavaScript/Guide/Typed_arrays)