Кратко
Секция статьи "Кратко"FormData
— это специальная коллекция данных, которая позволяет передавать данные в виде пар
на сервер при помощи fetch
или XMLHttpRequest
. При этом используется точно такой же формат данных, какой использует тег <form>
с типом кодирования "multipart
. Поэтому, значения в FormData
, как и у обычной HTML формы, могут быть только строками или файлами.
Пример
Секция статьи "Пример"Предположим, что мы пишем функцию, которая отправляет на сервер два поля name
и email
. Значения полей она получает через аргументы:
async function sendData(name, email) { const data = new FormData() data.append("name", name) data.append("email", email) return await fetch('/api/subscribe/', { method: "POST", body: data, })}
async function sendData(name, email) { const data = new FormData() data.append("name", name) data.append("email", email) return await fetch('/api/subscribe/', { method: "POST", body: data, }) }
Данные отправляются на сервер с помощью объекта FormData
. Мы используем метод append
чтобы добавить значения, а затем передаём полученный объект функции fetch
.
Как пишется
Секция статьи "Как пишется"Для работы с FormData
сначала с помощью конструктора new
создаётся объект этого типа: const form
. Затем у полученного объекта можно вызывать методы.
Основные методы для работы с FormData
:
append
— добавляет значение для ключа с сохранением предыдущих значений;( ключ , значение ) set
— устанавливает значение для ключа, перезаписывая предыдущие значения;( ключ , значение ) get
— возвращает первое значение ключа;( ключ ) getAll
— возвращает все значения ключа;( ключ ) has
— проверяет наличие переданного ключа;( ключ ) entries
— возвращает итератор пар( )
;[ ключ , значение ] values
— возвращает итератор всех значений коллекции;( ) keys
— возвращает итератор всех ключей коллекции;( ) delete
— удаляет конкретное значение;( ключ )
Как понять
Секция статьи "Как понять"При отправке данных на сервер и сервер и клиент должны понимать друг друга, то есть использовали понятные обоим способы кодирования и декодирования данных. Таких способов существует большое количество и FormData
позволяет работать с одним из них — "multipart
.
FormData
похожа на коллекцию Map
— предоставляет удобные методы для добавления и удаления данных. Но если передать её в качестве тела запроса при вызове fetch
(как в примере выше), данные «под капотом» будут преобразованы в нужный формат, а HTTP-заголовку Content
будет присвоено значение "multipart
, чтобы сервер знал, что именно с этим форматом ему предстоит работать.
FormData
является отражением данных обычной HTML-формы с атрибутом enctype
, поэтому пример выше можно представить следующим образом без JavaScript:
<form method="post" action="/api/subscribe/" enctype="multipart/form-data"> <input type="text" name="name" value="" /> <input type="email" name="email" value="" /> <button type="submit">Отправить</button></form>
<form method="post" action="/api/subscribe/" enctype="multipart/form-data"> <input type="text" name="name" value="" /> <input type="email" name="email" value="" /> <button type="submit">Отправить</button> </form>
Когда выбирать
Секция статьи "Когда выбирать"Существует несколько самых популярных способов кодирования данных для отправки на сервер: "application
, "multipart
и "application
. Иногда бывает так, что сервер поддерживает только какой-то определённый способ. Тогда выбирать не приходится. Но чаще всего современные решения на бэкенде поддерживают несколько способов, поэтому выбирать нужно в зависимости от задачи.
"application
— способ, который используют HTML-формы по умолчанию. Из-за особенностей преобразования, этот способ плохо подходит для больших объёмов данных, в особенности, файлов или строк с большим количеством символов не из ASCII таблицы (например, символы русского алфавита)./ x - www - form - urlencoded" "application
— достаточно популярный формат из-за широкого распространения JSON как формата обмена данными. Из плюсов - поддерживает вложенные структуры, поэтому можно в одном запросе отправить, например, целый объект с данными. Однако, чтобы отправить файл при помощи этого формата, необходимо файл дополнительно закодировать в строку каким-нибудь алгоритмом, например Base64. Причём на сервере нужно декодировать эти данные обратно./ json" "multipart
— удобный способ для загрузки файлов, оптимален с точки зрения размера закодированных данных, но в качестве значений может хранить только строки или файлы./ form - data"
Поэтому, лучше всего использовать FormData
для отправки файлов на сервер или когда поддержка только строковых данных не является проблемой. Дополнительно, при создании FormData
можно передать DOM-элемент формы (будет рассмотрено ниже), и коллекция вытащит из этой формы все данные. Поэтому, если стоит задача отравить данные какой-либо формы, FormData
позволит сделать это с минимумом кода.
Создание FormData
Секция статьи "Создание FormData" Создать новый пустой объект FormData
можно с помощью конструктора:
const data = new FormData()
const data = new FormData()
Также, конструктор может принимать в качестве аргумента DOM-элемент формы, в этом случае FormData
запишет текущие значения полей формы:
<form id="user-form"> <input type="text" name="name" value="Аня" /> <input type="text" name="language" value="JavaScript" /></form>
<form id="user-form"> <input type="text" name="name" value="Аня" /> <input type="text" name="language" value="JavaScript" /> </form>
const form = document.querySelector("#user-form")const data = new FormData(form)for (let [key, value] of data) { console.log(`${key} - ${value}`)}// "name - Аня"// "language - JavaScript"
const form = document.querySelector("#user-form") const data = new FormData(form) for (let [key, value] of data) { console.log(`${key} - ${value}`) } // "name - Аня" // "language - JavaScript"
Работа с коллекцией
Секция статьи "Работа с коллекцией"Для добавления данных в коллекцию есть метод append
:
const data = new FormData()data.append("name", "Вася")
const data = new FormData() data.append("name", "Вася")
Теперь в коллекции появилось одно значение с ключом name
и значением "Вася".
После выполнения этого кода, в коллекции будет два значения ("Вася" и "Лена") для одного ключа name
:
const data = new FormData()data.append("name", "Вася")data.append("name", "Лена")
const data = new FormData() data.append("name", "Вася") data.append("name", "Лена")
FormData
поддерживает ещё один метод для записи данных: set
. В отличие от append
, он перезапишет старые данные для переданного ключа, если они были:
const data = new FormData()data.set("name", "Вася")data.set("name", "Лена")
const data = new FormData() data.set("name", "Вася") data.set("name", "Лена")
В коллекции у ключа name
будет одно значение "Лена", потому что прошлое значение было перезаписано.
Вот пример такого поведения. Записываем число 30
, но фактически записывается строка "30"
:
const data = new FormData()data.append("age", 30)console.log(data.get("age") === 30);// falseconsole.log(data.get("age") === "30");// trueconsole.log(typeof data.get("age"));// "string"
const data = new FormData() data.append("age", 30) console.log(data.get("age") === 30); // false console.log(data.get("age") === "30"); // true console.log(typeof data.get("age")); // "string"
Для получения записанных значений есть два метода: get
и getAll
. get
вернёт первое значение для ключа или null
, если для указанного ключа значений не было:
const data = new FormData()console.log(data.get("name"))// nulldata.append("name", "Вася")data.append("name", "Лена")console.log(data.get("name"))// "Вася"
const data = new FormData() console.log(data.get("name")) // null data.append("name", "Вася") data.append("name", "Лена") console.log(data.get("name")) // "Вася"
В примере выше второе значение "Лена" при помощи метода get
недоступно, потому что он всегда возвращает только первое значение. Поэтому, чтобы получить все значения, на помощь приходит getAll
. Он всегда возвращает массив значений для указанного ключа. Если значений не было, возвращаемый массив будет пустой:
const data = new FormData()console.log(data.getAll("name"))// []data.append("name", "Вася")data.append("name", "Лена")console.log(data.getAll("name"))// ["Вася", "Лена"]
const data = new FormData() console.log(data.getAll("name")) // [] data.append("name", "Вася") data.append("name", "Лена") console.log(data.getAll("name")) // ["Вася", "Лена"]
Чтобы проверить, есть ли в коллекции данные для определённого ключа, есть метод has
, он вернёт true или false:
const data = new FormData()console.log(data.has("name"))// falsedata.append("name", "Вася")console.log(data.has("name"))// true
const data = new FormData() console.log(data.has("name")) // false data.append("name", "Вася") console.log(data.has("name")) // true
Для удаления значений для определённого ключа можно воспользоваться методом delete
. Важно помнить, что если у указанного ключа несколько значений, то удалятся все значения:
const data = new FormData()data.append("name", "Вася")data.append("name", "Лена")data.delete("name")
const data = new FormData() data.append("name", "Вася") data.append("name", "Лена") data.delete("name")
После выполнения кода, коллекция снова будет пустая. Мы удалили ключ целиком, поэтому оба значения по этому ключу исчезли.
Обход значений
Секция статьи "Обход значений"FormData
предоставляет встроенный итератор для обхода значений:
const data = new FormData()data.append("name", "Вася")data.append("name", "Лена")data.append("language", "JavaScript")for (let [key, value] of data) { console.log(`${key} - ${value}`)}// name - Вася// name - Лена// language - JavaScript
const data = new FormData() data.append("name", "Вася") data.append("name", "Лена") data.append("language", "JavaScript") for (let [key, value] of data) { console.log(`${key} - ${value}`) } // name - Вася // name - Лена // language - JavaScript
Тот же итератор доступен при помощи метода entries
. Обратите внимание, что каждый элемент итератора — массив из двух элементов. Первый элемент — ключ, а второй — значение.
В дополнение к этому, FormData
предоставляет два других итератора: только ключей при помощи метода keys
и только значений при помощи values
. Каждый ключ при перечислении ключей будет появляться ровно столько раз, сколько значений он содержит:
const data = new FormData()data.append("name", "Вася")data.append("name", "Лена")data.append("language", "JavaScript")console.log("Проходимся по значениям:")for (let value of data.values()) { console.log(value)}// "Вася"// "Лена"// "JavaScript"console.log("Проходимся по ключам:")for (let key of data.keys()) { console.log(key)}// "name"// "name" <-- ключ появился второй раз, потому что содержит два значения// "language"
const data = new FormData() data.append("name", "Вася") data.append("name", "Лена") data.append("language", "JavaScript") console.log("Проходимся по значениям:") for (let value of data.values()) { console.log(value) } // "Вася" // "Лена" // "JavaScript" console.log("Проходимся по ключам:") for (let key of data.keys()) { console.log(key) } // "name" // "name" <-- ключ появился второй раз, потому что содержит два значения // "language"
На практике
Секция статьи "На практике"Сильной стороной FormData
является загрузка файлов на сервер. Если при использовании "application
файлы необходимо дополнительно кодировать каким-то способом, чтобы привести к строке (и точно так-же декодировать на сервере), то FormData
умеет это делать «из коробки».
Например, если мы хотим после выбора файла сразу же загрузить его на сервер, то нам понадобится следующий HTML:
<input id="file-input" type="file" />
<input id="file-input" type="file" />
А для отправки на сервер просто добавим файл в объект FormData
и отправим его на сервер:
// Объявляем функцию загрузки файлаfunction sendFile(file) { const data = new FormData(); // Добавляем файл data.append("document", file) return fetch('/api/upload/', { method: "POST", body: data, })}const fileInput = document.querySelector("#file-input")fileInput.addEventListener("change", (event) => { // Получаем файл. Обратите внимание, что файлов может быть несколько если у инпута стоит атрибут `multiple` const file = event.target.files[0] // Отправляем файл на сервер при помощи созданной функции sendFile(file) // Очищаем текущее значение инпута. Если этого не делать, то при ошибке загрузки, повторный выбор того же файла не вызовет событие _change_ event.target.value = null})
// Объявляем функцию загрузки файла function sendFile(file) { const data = new FormData(); // Добавляем файл data.append("document", file) return fetch('/api/upload/', { method: "POST", body: data, }) } const fileInput = document.querySelector("#file-input") fileInput.addEventListener("change", (event) => { // Получаем файл. Обратите внимание, что файлов может быть несколько если у инпута стоит атрибут `multiple` const file = event.target.files[0] // Отправляем файл на сервер при помощи созданной функции sendFile(file) // Очищаем текущее значение инпута. Если этого не делать, то при ошибке загрузки, повторный выбор того же файла не вызовет событие _change_ event.target.value = null })