Как преобразовать FormData (объект HTML5) в JSON


112

Как преобразовать объект HTML5 FormData в JSON? Без Jquery и обработки вложенных свойств в FormData как объекта.


2
Что ты пытаешься сделать? Помогает JSON.stringify()? Может ты попробуешь исправить то, что может быть сделано другим способом?
Justinas

Возможный дубликат Convert JS object to JSON string
Лиам

4
Не дублируется, поскольку я не хочу преобразовывать объект javascript в json и не хочу использовать Jquery.serialize ()
Леонардо

Ответы:


158

Вы также можете использовать непосредственно forEachна FormDataобъекте:

var object = {};
formData.forEach(function(value, key){
    object[key] = value;
});
var json = JSON.stringify(object);

ОБНОВИТЬ:

И для тех, кто предпочитает такое же решение со стрелочными функциями ES6 :

var object = {};
formData.forEach((value, key) => object[key] = value);
var json = JSON.stringify(object);

ОБНОВЛЕНИЕ 2:

А для тех, кому нужна поддержка списков с множественным выбором или других элементов формы с несколькими значениями (поскольку под ответом по этой проблеме так много комментариев, я добавлю возможное решение) :

var object = {};
formData.forEach((value, key) => {
    // Reflect.has in favor of: object.hasOwnProperty(key)
    if(!Reflect.has(object, key)){
        object[key] = value;
        return;
    }
    if(!Array.isArray(object[key])){
        object[key] = [object[key]];    
    }
    object[key].push(value);
});
var json = JSON.stringify(object);

Здесь Fiddle демонстрирует использование этого метода с простым списком множественного выбора.

ОБНОВЛЕНИЕ 3:

В качестве побочного примечания для тех, кто здесь оказался, в случае, если целью преобразования данных формы в json является их отправка через HTTP-запрос XML на сервер, вы можете отправить FormDataобъект напрямую, не преобразовывая его. Вот так просто:

var request = new XMLHttpRequest();
request.open("POST", "http://example.com/submitform.php");
request.send(formData);

См. Также Использование объектов FormData в MDN для справки :

ОБНОВЛЕНИЕ 4:

Как упоминалось в одном из комментариев ниже моего ответа, stringifyметод JSON не будет работать из коробки для всех типов объектов. Для получения дополнительной информации о том, какие типы поддерживаются, я хотел бы обратиться к разделу Описание в документации MDNJSON.stringify .

В описании также упоминается, что:

Если значение имеет метод toJSON (), он отвечает за определение того, какие данные будут сериализованы.

Это означает, что вы можете предоставить свой собственный toJSONметод сериализации с логикой для сериализации ваших пользовательских объектов. Таким образом вы можете быстро и легко создать поддержку сериализации для более сложных деревьев объектов.


1
Как упоминалось в ответе @TomasPrado, убедитесь, что вам не нужна поддержка IE11.
Уилт

4
Это не работает с элементами формы с множественным выбором, поскольку они используют один и тот же ключ, вы в конечном итоге перезаписываете значения, возвращая только последний выбранный элемент
Шон

1
@Sean Я дал ответ, который работает с несколькими значениями для одного <SELECT MULTIPLE>и того <INPUT type="checkbox">же имени, путем преобразования значения в массив.
некоторые

formData не является серией. как можно итерировать? принял ответ, но что-то упустил.
Нури ИЛМАЗ

4
Если нет необходимости в множественном выборе и т. Д., Ответ JSON.stringify(Object.fromEntries(formData));будет намного лучше
Том Стикель

112

В 2019 году такая задача стала сверхлегкой.

JSON.stringify(Object.fromEntries(formData));

Object.fromEntries: Поддерживается в Chrome 73+, Firefox 63+, Safari 12.1.


2
Пришел сюда, чтобы опубликовать это,
Марцин

14
Это не работает правильно с формами, которые имеют несколько полей с одинаковым именем.
JukkaP

3
Это 2020 и это не обрабатывать несколько выбранных значений <select multiple>или <input type="checkbox"> 😞
какой - то

4
Лучше использовать formData.entries:JSON.stringify(Object.fromEntries(formData.entries()));
Кохвер

22

Вот способ сделать это в более функциональном стиле без использования библиотеки.

Array.from(formData.entries()).reduce((memo, pair) => ({
  ...memo,
  [pair[0]]: pair[1],
}), {});

Пример:

document.getElementById('foobar').addEventListener('submit', (e) => {
  e.preventDefault();

  const formData = new FormData(e.target);
  const data = Array.from(formData.entries()).reduce((memo, pair) => ({
    ...memo,
    [pair[0]]: pair[1],
  }), {});
  document.getElementById('output').innerHTML = JSON.stringify(data);
});
<form id='foobar'>
  <input name='baz' />
  <input type='submit' />
</form>

<pre id='output'>Input some value and submit</pre>


1
Лучший ответ здесь. Спасибо :)
Chunky Chunk

4
Мне очень нравится этот ответ, но я все еще не обрабатываю несколько элементов. Я опубликовал новый ответ, основанный на этом, для обработки этих случаев.
CarlosH.

10

Если у вас есть несколько записей с одним и тем же именем, например, если вы используете <SELECT multiple>или имеете несколько записей с <INPUT type="checkbox">одинаковым именем, вам нужно позаботиться об этом и создать массив значений. В противном случае вы получите только последнее выбранное значение.

Вот современный вариант ES6:

function formToJSON( elem ) {
  let output = {};
  new FormData( elem ).forEach(
    ( value, key ) => {
      // Check if property already exist
      if ( Object.prototype.hasOwnProperty.call( output, key ) ) {
        let current = output[ key ];
        if ( !Array.isArray( current ) ) {
          // If it's not an array, convert it to an array.
          current = output[ key ] = [ current ];
        }
        current.push( value ); // Add the new value to the array.
      } else {
        output[ key ] = value;
      }
    }
  );
  return JSON.stringify( output );
}

Чуть старше код (но не поддерживается IE11, так как он не поддерживает ForEachили entriesна FormData)

function formToJSON( elem ) {
  var current, entries, item, key, output, value;
  output = {};
  entries = new FormData( elem ).entries();
  // Iterate over values, and assign to item.
  while ( item = entries.next().value )
    {
      // assign to variables to make the code more readable.
      key = item[0];
      value = item[1];
      // Check if key already exist
      if (Object.prototype.hasOwnProperty.call( output, key)) {
        current = output[ key ];
        if ( !Array.isArray( current ) ) {
          // If it's not an array, convert it to an array.
          current = output[ key ] = [ current ];
        }
        current.push( value ); // Add the new value to the array.
      } else {
        output[ key ] = value;
      }
    }
    return JSON.stringify( output );
  }

7

Вы можете добиться этого с помощью объекта FormData () . Этот объект FormData будет заполнен текущими ключами / значениями формы с использованием свойства name каждого элемента для ключей и их отправленного значения для значений. Он также будет кодировать входное содержимое файла.

Пример:

var myForm = document.getElementById('myForm');
myForm.addEventListener('submit', function(event)
{
    event.preventDefault();
    var formData = new FormData(myForm),
        result = {};

    for (var entry of formData.entries())
    {
        result[entry[0]] = entry[1];
    }
    result = JSON.stringify(result)
    console.log(result);

});

Это не создает json
Лиам

1
@Liam Вы пробовали это с элементами формы? И дайте мне знать, почему он не создает объект JSON?
GiriB 03

1
Не существует такого понятия, как объект json. Json - это строковая нотация
Лиам

1
@Liam После создания объекта они могут использовать JSON.stringify (результат). И я отредактировал свой ответ. Пожалуйста, проверь это. И отозвать голосование "против".
GiriB 03

1
Вы также можете сделать выражение for of более выразительным, если используете ES6: for (const [key, value] of formData.entries())
Тедди

7

Простая в использовании функция

Я создал для этого функцию

function FormDataToJSON(FormElement){    
    var formData = new FormData(FormElement);
    var ConvertedJSON= {};
    for (const [key, value]  of formData.entries())
    {
        ConvertedJSON[key] = value;
    }

    return ConvertedJSON
}

Пример использования

var ReceivedJSON = FormDataToJSON(document.getElementById('FormId');)

В этом коде я создал пустую переменную JSON, используя forцикл, который я использовалkey от объекта formData до ключей JSON в каждой итерации.

Вы найдете этот код в моей библиотеке JS на GitHub. Предложите мне, если он нуждается в улучшении. Я разместил код здесь https://github.com/alijamal14/Utilities/blob/master/Utilities.js


1
@zuluk объяснил спасибо
Али Джамал

Это не обрабатывает несколько выбранных значений <select multiple>или <input type="checkbox">.
некоторые

5

Этому посту уже год ... но мне очень нравится ответ ES6 @dzuc. Однако он неполный, поскольку не может обрабатывать несколько вариантов выбора или флажков. На это уже указывалось, и предлагались кодовые решения. Я считаю их тяжелыми и не оптимизированными. Поэтому я написал 2 версии на основе @dzuc для обработки этих случаев:

  • Для форм в стиле ASP, где названия нескольких элементов могут просто повторяться.
let r=Array.from(fd).reduce(
  (o , [k,v]) => (
     (!o[k])
     ? {...o , [k] : v}
     : {...o , [k] : [...o[k] , v]}
   )
   ,{}
);
let obj=JSON.stringify(r);

Однострочная версия Hotshot:

Array.from(fd).reduce((o,[k,v])=>((!o[k])?{...o,[k]:v}:{...o,[k]:[...o[k],v]}),{});
  • Для форм в стиле PHP, в которых несколько имен элементов должны иметь []суффикс.
let r=Array.from(fd).reduce(
  (o , [k,v]) => (
    (k.split('[').length>1)
    ? (k=k.split('[')[0]
      , (!o[k])
      ? {...o , [k] : [v]}
      : {...o , [k] : [...o[k] , v ]}
    )
    : {...o , [k] : v}
  )
  ,{}
);
let obj=JSON.stringify(r);

Однострочная версия Hotshot:

Array.from(fd).reduce((o,[k,v])=>((k.split('[').length>1)?(k=k.split('[')[0],(!o[k])?{...o,[k]:[v]}:{...o,[k]:[...o[k],v]}):{...o,[k]:v}),{});
  • Расширение формы PHP, поддерживающее многоуровневые массивы.

Поскольку в прошлый раз я писал предыдущий второй кейс, на работе случился случай, когда в форме PHP есть флажки на многоуровневых уровнях. Я написал новый случай, чтобы подтвердить предыдущий и этот. Я создал фрагмент, чтобы лучше продемонстрировать этот случай, результат отображается на консоли для этой демонстрации, измените его по своему усмотрению. Пытался оптимизировать его, насколько мог, без ущерба для производительности, однако он в некоторой степени ухудшил удобочитаемость. Преимущество заключается в том, что массивы являются объектами, а переменные, указывающие на массивы, хранятся как ссылки. Нет уж, будь моим гостем.

let nosubmit = (e) => {
  e.preventDefault();
  const f = Array.from(new FormData(e.target));
  const obj = f.reduce((o, [k, v]) => {
    let a = v,
      b, i,
      m = k.split('['),
      n = m[0],
      l = m.length;
    if (l > 1) {
      a = b = o[n] || [];
      for (i = 1; i < l; i++) {
        m[i] = (m[i].split(']')[0] || b.length) * 1;
        b = b[m[i]] = ((i + 1) == l) ? v : b[m[i]] || [];
      }
    }
    return { ...o, [n]: a };
  }, {});
  console.log(obj);
}
document.querySelector('#theform').addEventListener('submit', nosubmit, {capture: true});
<h1>Multilevel Form</h1>
<form action="#" method="POST" enctype="multipart/form-data" id="theform">
  <input type="hidden" name="_id" value="93242" />
  <input type="hidden" name="_fid" value="45c0ec96929bc0d39a904ab5c7af70ef" />
  <label>Select:
    <select name="uselect">
      <option value="A">A</option>
      <option value="B">B</option>
      <option value="C">C</option>
    </select>
  </label>
  <br /><br />
  <label>Checkboxes one level:<br/>
    <input name="c1[]" type="checkbox" checked value="1"/>v1 
    <input name="c1[]" type="checkbox" checked value="2"/>v2
    <input name="c1[]" type="checkbox" checked value="3"/>v3
  </label>
  <br /><br />
  <label>Checkboxes two levels:<br/>
    <input name="c2[0][]" type="checkbox" checked value="4"/>0 v4 
    <input name="c2[0][]" type="checkbox" checked value="5"/>0 v5
    <input name="c2[0][]" type="checkbox" checked value="6"/>0 v6
    <br/>
    <input name="c2[1][]" type="checkbox" checked value="7"/>1 v7 
    <input name="c2[1][]" type="checkbox" checked value="8"/>1 v8
    <input name="c2[1][]" type="checkbox" checked value="9"/>1 v9
  </label>
  <br /><br />
  <label>Radios:
    <input type="radio" name="uradio" value="yes">YES
    <input type="radio" name="uradio" checked value="no">NO
  </label>
  <br /><br />
  <input type="submit" value="Submit" />
</form>


2
Array.from(fd).reduce((obj, [k, v]) => ({...obj, [k]: v}), {});hotshot version es2018
nackjicholson

1
@nackjicholson Да, вы правы, исключение .entries () и элемента [k, v] упрощает код. Я перепишу код, чтобы включить эти улучшения. Однако ваш по-прежнему будет перезаписывать повторяющиеся значения.
CarlosH.

3
Хотя я ценю усилия - такой код абсурден. Никто не хочет смотреть на буквы для переменных и объектов, это не 1985 год.
Том Стикель

4

Метод FormData .entriesи for ofвыражение не поддерживаются в IE11 и Safari.

Вот более простая версия для поддержки Safari, Chrome, Firefox и Edge.

function formDataToJSON(formElement) {    
    var formData = new FormData(formElement),
        convertedJSON = {};

    formData.forEach(function(value, key) { 
        convertedJSON[key] = value;
    });

    return convertedJSON;
}

Предупреждение: этот ответ не работает в IE11.
FormData не имеет forEachметода в IE11.
Я все еще ищу окончательное решение для поддержки всех основных браузеров.


это потрясающе! мы должны поддерживать старые браузеры, а использование итератора не очень интуитивно понятно.
Питер Хокинс

4

Если вам нужна поддержка для сериализации вложенных полей, аналогично тому, как PHP обрабатывает поля формы, вы можете использовать следующую функцию

function update(data, keys, value) {
  if (keys.length === 0) {
    // Leaf node
    return value;
  }

  let key = keys.shift();
  if (!key) {
    data = data || [];
    if (Array.isArray(data)) {
      key = data.length;
    }
  }

  // Try converting key to a numeric value
  let index = +key;
  if (!isNaN(index)) {
    // We have a numeric index, make data a numeric array
    // This will not work if this is a associative array 
    // with numeric keys
    data = data || [];
    key = index;
  }
  
  // If none of the above matched, we have an associative array
  data = data || {};

  let val = update(data[key], keys, value);
  data[key] = val;

  return data;
}

function serializeForm(form) {
  return Array.from((new FormData(form)).entries())
    .reduce((data, [field, value]) => {
      let [_, prefix, keys] = field.match(/^([^\[]+)((?:\[[^\]]*\])*)/);

      if (keys) {
        keys = Array.from(keys.matchAll(/\[([^\]]*)\]/g), m => m[1]);
        value = update(data[prefix], keys, value);
      }
      data[prefix] = value;
      return data;
    }, {});
}

document.getElementById('output').textContent = JSON.stringify(serializeForm(document.getElementById('form')), null, 2);
<form id="form">
  <input name="field1" value="Field 1">
  <input name="field2[]" value="Field 21">
  <input name="field2[]" value="Field 22">
  <input name="field3[a]" value="Field 3a">
  <input name="field3[b]" value="Field 3b">
  <input name="field3[c]" value="Field 3c">
  <input name="field4[x][a]" value="Field xa">
  <input name="field4[x][b]" value="Field xb">
  <input name="field4[x][c]" value="Field xc">
  <input name="field4[y][a]" value="Field ya">
  <input name="field5[z][0]" value="Field z0">
  <input name="field5[z][]" value="Field z1">
  <input name="field6.z" value="Field 6Z0">
  <input name="field6.z" value="Field 6Z1">
</form>

<h2>Output</h2>
<pre id="output">
</pre>


1
Необходимо обновить значение по умолчанию для «данных» из массива «[]» до объекта «{}» в «обновлении функции (данные, ключи, значение) {», это устраняет проблему с пустым массивом.
Чинтан Матукия

Я также предлагаю добавить фильтр массива перед уменьшением, чтобы удалить пустые поля из конечного объекта
Эднилсон Майя

@ChintanMathukiya Maia Можете ли вы поделиться примером ввода, для которого у вас неожиданный результат?
Джойс Бабу,

@Ednilson Можете ли вы поделиться образцами данных, для которых вы видите пустой массив?
Джойс Бабу,

3

Если вы используете lodash, это можно сделать кратко с помощью fromPairs

import {fromPairs} from 'lodash';

const object = fromPairs(Array.from(formData.entries()));

2

Вы можете попробовать это

formDataToJSON($('#form_example'));

# Create a function to convert the serialize and convert the form data
# to JSON
# @param : $('#form_example');
# @return a JSON Stringify
function formDataToJSON(form) {
    let obj = {};
    let formData = form.serialize();
    let formArray = formData.split("&");

    for (inputData of formArray){
        let dataTmp = inputData.split('=');
        obj[dataTmp[0]] = dataTmp[1];
    }
    return JSON.stringify(obj);
}

2

Несмотря на то, что ответ от @dzuc уже очень хорош, вы можете использовать деструктуризацию массива (доступную в современных браузерах или в Babel), чтобы сделать его еще немного более элегантным:

// original version from @dzuc
const data = Array.from(formData.entries())
  .reduce((memo, pair) => ({
    ...memo,
    [pair[0]: pair[1],
  }), {})

// with array destructuring
const data = Array.from(formData.entries())
  .reduce((memo,[key, value]) => ({
    ...memo,
    [key]: value,
  }), {})

2

Оскорбительная однострочка!

Array.from(fd).reduce((obj, [k, v]) => ({...obj, [k]: v}), {});

Сегодня я узнал, что firefox поддерживает распространение объектов и деструктуризацию массивов!


1

Если следующие предметы соответствуют вашим потребностям, вам повезло:

  1. Вы хотите преобразовать массив массивов, таких как [['key','value1'], ['key2','value2'](например, то, что дает вам FormData), в объект key-> value, например{key1: 'value1', key2: 'value2'} и преобразовать его в строку JSON.
  2. Вы ориентируетесь на браузеры / устройства с последней версией интерпретатора ES6 или компилируете что-то вроде babel.
  3. Вам нужен крошечный способ добиться этого.

Вот код, который вам понадобится:

const data = new FormData(document.querySelector('form'));
const json = JSON.stringify(Array.from(data).reduce((o,[k,v])=>(o[k]=v,o),{}));

Надеюсь, это кому-то поможет.


1

Я не видел упоминаний о FormData.getAll методе .

Помимо возврата всех значений, связанных с данным ключом, из объекта FormData, становится очень просто использовать метод Object.fromEntries, как указано здесь другими.

var formData = new FormData(document.forms[0])

var obj = Object.fromEntries(
  Array.from(formData.keys()).map(key => [
    key, formData.getAll(key).length > 1 ? 
      formData.getAll(key) : formData.get(key)
  ])
)

Фрагмент в действии

var formData = new FormData(document.forms[0])

var obj = Object.fromEntries(Array.from(formData.keys()).map(key => [key, formData.getAll(key).length > 1 ? formData.getAll(key) : formData.get(key)]))

document.write(`<pre>${JSON.stringify(obj)}</pre>`)
<form action="#">
  <input name="name" value="Robinson" />
  <input name="items" value="Vin" />
  <input name="items" value="Fromage" />
  <select name="animals" multiple id="animals">
    <option value="tiger" selected>Tigre</option>
    <option value="turtle" selected>Tortue</option>
    <option value="monkey">Singe</option>
  </select>
</form>


1

Я думаю, что это самый простой способ получить formDataжелаемый результат от объекта FormData:

const jsonData = {};

for(const [key, value] of formData) {
    jsonData[key] = value;
}

0

Работал на меня

                var myForm = document.getElementById("form");
                var formData = new FormData(myForm),
                obj = {};
                for (var entry of formData.entries()){
                    obj[entry[0]] = entry[1];
                }
                console.log(obj);

Он не будет обрабатывать несколько выбранных значений <select multiple> или<input type="checkbox">
некоторые

0

В моем случае форма Data была данными, база огня ожидала объект, но данные содержат объект, а также все другие материалы, поэтому я попробовал data.value, это сработало !!!


0

Я опаздываю сюда. Однако я сделал простой метод, который проверяет тип ввода = "checkbox"

var formData = new FormData($form.get(0));
        var objectData = {};
        formData.forEach(function (value, key) {
            var updatedValue = value;
            if ($('input[name="' + key + '"]').attr("type") === "checkbox" && $('input[name="' + key + '"]').is(":checked")) {
                updatedValue = true; // we don't set false due to it is by default on HTML
            }
            objectData[key] = updatedValue;
        });
var jsonData = JSON.stringify(objectData);

Надеюсь, это поможет кому-то другому.


-1

Вы можете легко выполнить эту работу, не используя ничего особенного. Код, подобный приведенному ниже, будет достаточно.

var form = $(e.currentTarget);

var formData = objectifyForm(form);


function objectifyForm(formArray) {

    var returnArray = {};
    for (var i = 0; i < formArray[0].length; i++) {
        var name = formArray[0][i]['name'];
        if (name == "") continue;
        if (formArray[0][i]['type'] == "hidden" && returnArray[name] != undefined) continue;
        if ($(formArray[0][i]).attr("type") == "radio") {
            var radioInputs = $("[name='" + name + "']");
            var value = null;
            radioInputs.each(function (radio) {
                if ($(this)[0].checked == true) {
                    value = $(this).attr("id").split("_")[$(this).attr("id").split("_").length - 1];
                }
            });
            returnArray[name] = value;
        }
        else if ($(formArray[0][i]).attr("type") == "checkbox") {
            returnArray[name] = $(formArray[0][i])[0].checked;
        }
        else
            returnArray[name] = formArray[0][i]['value'];
    }



    return returnArray;
};
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.