如何将表单的所有元素转换为JavaScript对象?
我希望有一些方法可以自动从表单中构建JavaScript对象,而不必遍历每个元素。我不想要由返回的字符串,$('#formid').serialize();
也不希望由返回的映射$('#formid').serializeArray();
如何将表单的所有元素转换为JavaScript对象?
我希望有一些方法可以自动从表单中构建JavaScript对象,而不必遍历每个元素。我不想要由返回的字符串,$('#formid').serialize();
也不希望由返回的映射$('#formid').serializeArray();
Answers:
serializeArray
已经做到了。您只需要将数据整理成所需的格式:
function objectifyForm(formArray) {//serialize data function
var returnArray = {};
for (var i = 0; i < formArray.length; i++){
returnArray[formArray[i]['name']] = formArray[i]['value'];
}
return returnArray;
}
注意与真实输入同名的隐藏字段,因为它们会被覆盖。
$.map( $("#container :input"), function(n, i) { /* n.name and $(n).val() */ } );
如果需要包含禁用的元素,最好使用类似的东西。
foo[bar]
按预期处理-type输入,更不用说其他大多数输入名称变体了。在对这个问题的浅层解决方案感到非常沮丧之后,我最终编写了自己的jQuery插件-有关此问题的详细信息。
$ bower安装jquery-serialize-object
以下代码可以处理各种输入名称;并按您期望的方式处理它们。
例如:
<!-- All of these will work! -->
<input name="honey[badger]" value="a">
<input name="wombat[]" value="b">
<input name="hello[panda][]" value="c">
<input name="animals[0][name]" value="d">
<input name="animals[0][breed]" value="e">
<input name="crazy[1][][wonky]" value="f">
<input name="dream[as][vividly][as][you][can]" value="g">
// Output
{
"honey":{
"badger":"a"
},
"wombat":["b"],
"hello":{
"panda":["c"]
},
"animals":[
{
"name":"d",
"breed":"e"
}
],
"crazy":[
null,
[
{"wonky":"f"}
]
],
"dream":{
"as":{
"vividly":{
"as":{
"you":{
"can":"g"
}
}
}
}
}
}
$('#my-form').serializeObject();
(function($){
$.fn.serializeObject = function(){
var self = this,
json = {},
push_counters = {},
patterns = {
"validate": /^[a-zA-Z][a-zA-Z0-9_]*(?:\[(?:\d*|[a-zA-Z0-9_]+)\])*$/,
"key": /[a-zA-Z0-9_]+|(?=\[\])/g,
"push": /^$/,
"fixed": /^\d+$/,
"named": /^[a-zA-Z0-9_]+$/
};
this.build = function(base, key, value){
base[key] = value;
return base;
};
this.push_counter = function(key){
if(push_counters[key] === undefined){
push_counters[key] = 0;
}
return push_counters[key]++;
};
$.each($(this).serializeArray(), function(){
// Skip invalid keys
if(!patterns.validate.test(this.name)){
return;
}
var k,
keys = this.name.match(patterns.key),
merge = this.value,
reverse_key = this.name;
while((k = keys.pop()) !== undefined){
// Adjust reverse_key
reverse_key = reverse_key.replace(new RegExp("\\[" + k + "\\]$"), '');
// Push
if(k.match(patterns.push)){
merge = self.build([], self.push_counter(reverse_key), merge);
}
// Fixed
else if(k.match(patterns.fixed)){
merge = self.build([], k, merge);
}
// Named
else if(k.match(patterns.named)){
merge = self.build({}, k, merge);
}
}
json = $.extend(true, json, merge);
});
return json;
};
})(jQuery);
有什么问题:
var data = {};
$(".form-selector").serializeArray().map(function(x){data[x.name] = x.value;});
.each
而不是.map
???
var form = {}; $.each($(this).serializeArray(), function (i, field) { form[field.name] = field.value || ""; });
$(this).serializeArray().reduce(function(m,o){ m[o.name] = o.value; return m;}, {})
Tobias Cohen解决方案的固定版本。此代码可以正确处理伪造的值,例如0
和''
。
jQuery.fn.serializeObject = function() {
var arrayData, objectData;
arrayData = this.serializeArray();
objectData = {};
$.each(arrayData, function() {
var value;
if (this.value != null) {
value = this.value;
} else {
value = '';
}
if (objectData[this.name] != null) {
if (!objectData[this.name].push) {
objectData[this.name] = [objectData[this.name]];
}
objectData[this.name].push(value);
} else {
objectData[this.name] = value;
}
});
return objectData;
};
还有一个CoffeeScript版本,以方便您进行编码:
jQuery.fn.serializeObject = ->
arrayData = @serializeArray()
objectData = {}
$.each arrayData, ->
if @value?
value = @value
else
value = ''
if objectData[@name]?
unless objectData[@name].push
objectData[@name] = [objectData[@name]]
objectData[@name].push value
else
objectData[@name] = value
return objectData
value = @value ? ''
keyMap
和以下行:key = if keyMap? then keyMap(@name) else @name
。现在,您可以传递类似的映射功能(name) -> name.match(/\[([^\]]+)]/)[1]
。再一个需要改变所有后续@name
到key
,当然
post
,您可以做$('form').serializeObject().post
。无需花哨的映射。
if (!objectData[this.name].push)
??
objectData[this.name]
具有push方法(大概是数组)。如果它是一个数组,它将推入该值;如果它不是一个数组,则将其转换为一个数组,以便将具有相同键的多个值组合到一个数组中。
我喜欢使用Array.prototype.reduce
它,因为它是单线的,并且不依赖于Underscore.js等:
$('#formid').serializeArray()
.reduce(function(a, x) { a[x.name] = x.value; return a; }, {});
这类似于使用的答案Array.prototype.map
,但是您不需要使用其他对象变量来使范围杂乱无章。一站式购物。
重要说明:具有具有重复name
属性的输入的表单是有效的HTML,实际上是一种常用方法。在这种情况下,在此线程中使用任何答案都是不合适的(因为对象键必须是唯一的)。
array.prototype.reduce()
是,它是ECMAScript 5规范的一部分,在IE8中不可用。因此,如果您需要IE8支持,则将要完全使用polyfill或其他解决方案。
所有这些答案对我而言似乎都是如此。为了简单起见,有话要说。只要您所有的表单输入都设置了name属性,它就应该只对jim dandy有效。
$('form.myform').submit(function () { var $this = $(this) , viewArr = $this.serializeArray() , view = {}; for (var i in viewArr) { view[viewArr[i].name] = viewArr[i].value; } //Do stuff with view object here (e.g. JSON.stringify?) });
如果不检查每个元素,实际上是没有办法做到的。您真正想知道的是“其他人是否已经编写了将表单转换为JSON对象的方法?” 类似于以下内容的方法应该起作用-请注意,它只会为您提供通过POST返回的表单元素(必须具有名称)。这未经测试。
function formToJSON( selector )
{
var form = {};
$(selector).find(':input[name]:enabled').each( function() {
var self = $(this);
var name = self.attr('name');
if (form[name]) {
form[name] = form[name] + ',' + self.val();
}
else {
form[name] = self.val();
}
});
return form;
}
如果您使用的是Underscore.js,则可以使用相对简洁的代码:
_.object(_.map($('#myform').serializeArray(), _.values))
我检查了所有其他答案是否有问题,如果输入名称是一个数组,例如name[key]
,则应该这样生成:
name:{ key : value }
例如:如果您的HTML表单类似于以下表单:
<form>
<input name="name" value="value" >
<input name="name1[key1]" value="value1" >
<input name="name2[key2]" value="value2" >
<input name="name3[key3]" value="value3" >
</form>
但是它应该像下面的JSON一样生成,并且不会与其他所有答案一样成为以下对象。因此,如果有人想带来类似以下JSON的内容,请尝试以下JS代码。
{
name : 'value',
name1 : { key1 : 'value1' },
name2 : { key2 : 'value2' },
name3 : { key2 : 'value2' }
}
$.fn.getForm2obj = function() {
var _ = {};
$.map(this.serializeArray(), function(n) {
const keys = n.name.match(/[a-zA-Z0-9_]+|(?=\[\])/g);
if (keys.length > 1) {
let tmp = _;
pop = keys.pop();
for (let i = 0; i < keys.length, j = keys[i]; i++) {
tmp[j] = (!tmp[j] ? (pop == '') ? [] : {} : tmp[j]), tmp = tmp[j];
}
if (pop == '') tmp = (!Array.isArray(tmp) ? [] : tmp), tmp.push(n.value);
else tmp[pop] = n.value;
} else _[keys.pop()] = n.value;
});
return _;
}
console.log($('form').getForm2obj());
$('form input').change(function() {
console.clear();
console.log($('form').getForm2obj());
});
<script src="https://code.jquery.com/jquery-3.2.1.min.js"></script>
<form>
<input name="name" value="value">
<input type="checkbox" name="name1[]" value="1" checked="checked">1
<input type="checkbox" name="name1[]" value="2">2
<input type="checkbox" name="name1[]" value="3">3<br>
<input type="radio" name="gender" value="male" checked="checked">male
<input type="radio" name="gender" value="female"> female
<input name="name2[key1]" value="value1">
<input name="one[another][another_one]" value="value4">
<input name="name3[1][name]" value="value4">
<input name="name3[2][name]" value="value4">
<input name="[]" value="value5">
</form>
eval
是因为不应以这种方式使用,因为eval
当以这种方式使用时,它们会提倡极其不可靠的,有漏洞的,不良的以及可能不安全的做法。
this.c = function(k,v){ eval("c = typeof "+k+";"); if(c == 'undefined') _t.b(k,v);}
简短而又不能解释。经验不足的开发人员只会在不了解其原因和工作方式的情况下复制它。
eval()
从代码中删除了它。
好的,我知道这已经得到了高度评价,但是最近又提出了另一个类似的问题,我也直接针对这个问题。我也想提供我的解决方案,因为它比公认的解决方案具有优势:您可以包括禁用的表单元素(有时很重要,具体取决于您的UI的功能)
这是我对另一个SO问题的回答:
最初,我们使用的是jQuery serializeArray()
方法,但是其中不包括被禁用的表单元素。我们通常会禁用与页面上其他源“同步”的表单元素,但是我们仍然需要将数据包含在序列化对象中。所以,serializeArray()
是的。我们使用:input
选择器来获取给定容器中的所有输入元素(已启用和已禁用),然后$.map()
创建我们的对象。
var inputs = $("#container :input");
var obj = $.map(inputs, function(n, i)
{
var o = {};
o[n.name] = $(n).val();
return o;
});
console.log(obj);
请注意,要使此功能起作用,您的每个输入都需要一个name
属性,该属性将成为结果对象的属性的名称。
实际上,这与我们使用的内容略有不同。我们需要创建一个结构为.NET IDictionary的对象,因此我们使用了以下方法:(如果有用,我在这里提供它)
var obj = $.map(inputs, function(n, i)
{
return { Key: n.name, Value: $(n).val() };
});
console.log(obj);
我喜欢这两种解决方案,因为它们是$.map()
函数的简单用法,并且您可以完全控制选择器(因此,最终将哪些元素包括在结果对象中)。另外,不需要额外的插件。普通的旧jQuery。
map
这样的方法创建具有单个属性的对象数组,它不会将所有属性都折叠为一个对象。
此函数应处理多维数组以及具有相同名称的多个元素。
到目前为止,我已经使用了几年:
jQuery.fn.serializeJSON=function() {
var json = {};
jQuery.map(jQuery(this).serializeArray(), function(n, i) {
var _ = n.name.indexOf('[');
if (_ > -1) {
var o = json;
_name = n.name.replace(/\]/gi, '').split('[');
for (var i=0, len=_name.length; i<len; i++) {
if (i == len-1) {
if (o[_name[i]]) {
if (typeof o[_name[i]] == 'string') {
o[_name[i]] = [o[_name[i]]];
}
o[_name[i]].push(n.value);
}
else o[_name[i]] = n.value || '';
}
else o = o[_name[i]] = o[_name[i]] || {};
}
}
else {
if (json[n.name] !== undefined) {
if (!json[n.name].push) {
json[n.name] = [json[n.name]];
}
json[n.name].push(n.value || '');
}
else json[n.name] = n.value || '';
}
});
return json;
};
采用:
function form_to_json (selector) {
var ary = $(selector).serializeArray();
var obj = {};
for (var a = 0; a < ary.length; a++) obj[ary[a].name] = ary[a].value;
return obj;
}
输出:
{"myfield": "myfield value", "passwordfield": "mypasswordvalue"}
我发现Tobias Cohen的代码有问题(我没有足够的要点直接对其进行评论),否则对我有用。如果您有两个具有相同名称的选择选项,且两个都具有value =“”,则原始代码将产生“ name”:“”而不是“ name”:[“”,“”]
我认为可以通过在第一个if条件上添加“ || o [this.name] ==”来解决此问题:
$.fn.serializeObject = function()
{
var o = {};
var a = this.serializeArray();
$.each(a, function() {
if (o[this.name] || o[this.name] == '') {
if (!o[this.name].push) {
o[this.name] = [o[this.name]];
}
o[this.name].push(this.value || '');
} else {
o[this.name] = this.value || '';
}
});
return o;
};
使用maček的解决方案,我对其进行了修改,使其可以与ASP.NET MVC以相同形式处理其嵌套/复杂对象的方式一起使用。您要做的就是将验证部分修改为:
"validate": /^[a-zA-Z][a-zA-Z0-9_]*((?:\[(?:\d*|[a-zA-Z0-9_]+)\])*(?:\.)[a-zA-Z][a-zA-Z0-9_]*)*$/,
这将匹配并正确映射名称如下的元素:
<input type="text" name="zooName" />
和
<input type="text" name="zooAnimals[0].name" />
对于jQuery,有一个插件可以做到这一点jquery.serializeJSON。我现在已经在一些项目中成功使用了它。它像一种魅力。
我更喜欢这种方法,因为:您不必重复两个集合,如果需要,可以使用除“名称”和“值”之外的其他东西,并且可以在将值存储到对象中之前对其进行清理(例如,如果您有不想存储的默认值)。
$.formObject = function($o) {
var o = {},
real_value = function($field) {
var val = $field.val() || "";
// additional cleaning here, if needed
return val;
};
if (typeof o != "object") {
$o = $(o);
}
$(":input[name]", $o).each(function(i, field) {
var $field = $(field),
name = $field.attr("name"),
value = real_value($field);
if (o[name]) {
if (!$.isArray(o[name])) {
o[name] = [o[name]];
}
o[name].push(value);
}
else {
o[name] = value;
}
});
return o;
}
像这样使用:
var obj = $.formObject($("#someForm"));
仅在Firefox中测试过。
将任何东西变成一个对象(未经单元测试)
<script type="text/javascript">
string = {};
string.repeat = function(string, count)
{
return new Array(count+1).join(string);
}
string.count = function(string)
{
var count = 0;
for (var i=1; i<arguments.length; i++)
{
var results = string.match(new RegExp(arguments[i], 'g'));
count += results ? results.length : 0;
}
return count;
}
array = {};
array.merge = function(arr1, arr2)
{
for (var i in arr2)
{
if (arr1[i] && typeof arr1[i] == 'object' && typeof arr2[i] == 'object')
arr1[i] = array.merge(arr1[i], arr2[i]);
else
arr1[i] = arr2[i]
}
return arr1;
}
array.print = function(obj)
{
var arr = [];
$.each(obj, function(key, val) {
var next = key + ": ";
next += $.isPlainObject(val) ? array.print(val) : val;
arr.push( next );
});
return "{ " + arr.join(", ") + " }";
}
node = {};
node.objectify = function(node, params)
{
if (!params)
params = {};
if (!params.selector)
params.selector = "*";
if (!params.key)
params.key = "name";
if (!params.value)
params.value = "value";
var o = {};
var indexes = {};
$(node).find(params.selector+"["+params.key+"]").each(function()
{
var name = $(this).attr(params.key),
value = $(this).attr(params.value);
var obj = $.parseJSON("{"+name.replace(/([^\[]*)/, function()
{
return '"'+arguments[1]+'"';
}).replace(/\[(.*?)\]/gi, function()
{
if (arguments[1].length == 0)
{
var index = arguments[3].substring(0, arguments[2]);
indexes[index] = indexes[index] !== undefined ? indexes[index]+1 : 0;
return ':{"'+indexes[index]+'"';
}
else
return ':{"'+escape(arguments[1])+'"';
})+':"'+value.replace(/[\\"]/gi, function()
{
return "\\"+arguments[0];
})+'"'+string.repeat('}', string.count(name, ']'))+"}");
o = array.merge(o, obj);
});
return o;
}
</script>
测试输出:
$(document).ready(function()
{
console.log(array.print(node.objectify($("form"), {})));
console.log(array.print(node.objectify($("form"), {selector: "select"})));
});
上
<form>
<input name='input[a]' type='text' value='text'/>
<select name='input[b]'>
<option>select</option>
</select>
<input name='otherinput[c][a]' value='a'/>
<input name='otherinput[c][]' value='b'/>
<input name='otherinput[d][b]' value='c'/>
<input name='otherinput[c][]' value='d'/>
<input type='hidden' name='anotherinput' value='hidden'/>
<input type='hidden' name='anotherinput' value='1'/>
<input type='submit' value='submit'/>
</form>
将产生:
{ input: { a: text, b: select }, otherinput: { c: { a: a, 0: b, 1: d }, d: { b: c } }, anotherinput: 1 }
{ input: { b: select } }
我发现所选解决方案有问题。
当使用具有基于数组的名称的表单时,jQuery serializeArray()函数实际上消失了。
我有一个PHP框架,该框架使用基于数组的字段名称,以允许将同一表单在多个视图中多次放置到同一页面上。将添加,编辑和删除都放在同一页面上而不会与表单模型冲突会很方便。
由于我想对表格进行序列化,而不必删除绝对的基本功能,因此我决定编写自己的seralizeArray():
var $vals = {};
$("#video_edit_form input").each(function(i){
var name = $(this).attr("name").replace(/editSingleForm\[/i, '');
name = name.replace(/\]/i, '');
switch($(this).attr("type")){
case "text":
$vals[name] = $(this).val();
break;
case "checkbox":
if($(this).attr("checked")){
$vals[name] = $(this).val();
}
break;
case "radio":
if($(this).attr("checked")){
$vals[name] = $(this).val();
}
break;
default:
break;
}
});
请注意:这也可以在表单commit()之外使用,因此,如果在其余代码中发生错误,则如果您将链接按钮放置为“保存更改”,则表单将不会提交。
还要注意,永远不要仅使用此函数来验证表单,以收集要发送到服务器端的数据以进行验证。使用这样的弱代码和大量分配的代码将导致XSS等。
我最近遇到了同样的问题,并提出了这个.toJSON jQuery插件,该插件将表单转换为具有相同结构的JSON对象。这对于要让用户在特定位置添加更多字段的动态生成的表单特别有用。
关键是您实际上可能希望构建一个表单,使其本身具有结构,因此,假设您要创建一个表单,以便用户在镇上插入他最喜欢的位置:您可以想象此表单表示一个<places>...</places>
包含以下内容的XML元素:用户喜欢的位置列表,因此喜欢一个<place>...</place>
元素列表,每个元素包含一个<name>...</name>
元素,一个<type>...</type>
元素,然后是一个元素列表,<activity>...</activity>
以表示您可以在这样的位置执行的活动。因此,您的XML结构将如下所示:
<places>
<place>
<name>Home</name>
<type>dwelling</type>
<activity>sleep</activity>
<activity>eat</activity>
<activity>watch TV</activity>
</place>
<place>...</place>
<place>...</place>
</places>
有了一个JSON对象来表示这个确切的结构,这将是多么酷,因此您可以:
好的,所以现在我们需要考虑表单如何表示XML文件。
当然,<form>
标签是root
,但随后我们有了该<place>
元素,它是一个容器,而不是数据元素本身,因此我们不能对其使用输入标签。
这是<fieldset>
标签派上用场的地方!我们将使用<fieldset>
标签来表示表单/ XML表示形式中的所有容器元素,从而得到如下结果:
<form name="places">
<fieldset name="place">
<input type="text" name="name"/>
<select name="type">
<option value="dwelling">Dwelling</option>
<option value="restoration">Restoration</option>
<option value="sport">Sport</option>
<option value="administrative">Administrative</option>
</select>
<input type="text" name="activity"/>
<input type="text" name="activity"/>
<input type="text" name="activity"/>
</fieldset>
</form>
正如您在这种形式中看到的那样,我们正在打破唯一名称的规则,但这是可以的,因为它们将被转换为元素数组,因此只能由它们在数组中的索引来引用。
此时,您可以看到name="array[]"
表单中没有类似的名称,并且所有内容都很漂亮,简单且具有语义。
现在,我们希望将此表单转换为如下所示的JSON对象:
{'places':{
'place':[
{
'name': 'Home',
'type': 'dwelling',
'activity':[
'sleep',
'eat',
'watch TV'
]
},
{...},
{...}
]
}}
为此,我在这里开发了这个jQuery插件,有人在此Code Review线程中进行了优化,如下所示:
$.fn.toJSO = function () {
var obj = {},
$kids = $(this).children('[name]');
if (!$kids.length) {
return $(this).val();
}
$kids.each(function () {
var $el = $(this),
name = $el.attr('name');
if ($el.siblings("[name=" + name + "]").length) {
if (!/radio|checkbox/i.test($el.attr('type')) || $el.prop('checked')) {
obj[name] = obj[name] || [];
obj[name].push($el.toJSO());
}
} else {
obj[name] = $el.toJSO();
}
});
return obj;
};
我也写了一篇博客文章来解释这一点。
这会将表单中的所有内容都转换为JSON(甚至是单选框和复选框),剩下要做的就是调用
$.post('script.php',('form').toJSO(), ...);
我知道有很多方法可以将表单转换为JSON对象,.serialize()
并且.serializeArray()
可以在大多数情况下正常工作,并且大多数情况下都打算使用这种方法,但是我认为将表单作为具有有意义名称的XML结构并将其转换为格式正确的JSON对象值得一试,如果您需要检索动态生成的表单数据,那么无需担心就可以添加同名输入标签,这一点非常有用。
我希望这可以帮助别人!
我自己将表单编码到多维JavaScript对象中,以便在生产中使用它。结果是https://github.com/serbanghita/formToObject.js。
另一个答案
document.addEventListener("DOMContentLoaded", function() {
setInterval(function() {
var form = document.getElementById('form') || document.querySelector('form[name="userprofile"]');
var json = Array.from(new FormData(form)).map(function(e,i) {this[e[0]]=e[1]; return this;}.bind({}))[0];
console.log(json)
document.querySelector('#asJSON').value = JSON.stringify(json);
}, 1000);
})
<form name="userprofile" id="form">
<p>Name <input type="text" name="firstname" value="John"/></p>
<p>Family name <input name="lastname" value="Smith"/></p>
<p>Work <input name="employment[name]" value="inc, Inc."/></p>
<p>Works since <input name="employment[since]" value="2017" /></p>
<p>Photo <input type="file" /></p>
<p>Send <input type="submit" /></p>
</form>
JSON: <textarea id="asJSON"></textarea>
FormData:https : //developer.mozilla.org/zh-CN/docs/Web/API/FormData
我喜欢samuels版本,但我相信它有一个小错误。通常,JSON发送为
{“ coreSKU”:“ PCGUYJS”,“ name_de”:“无论”,...
不像
[{“ coreSKU”:“ PCGUYJS”},{“ name_de”:“无论”},...
因此,IMO的功能应为:
App.toJson = function( selector ) {
var o = {};
$.map( $( selector ), function( n,i )
{
o[n.name] = $(n).val();
});
return o;
}
并将其包装在数据数组中(也是通常所期望的),最后将其作为字符串发送给App.stringify({data:App.toJson('#cropform:input'}})
对于快速,现代的解决方案,请使用JSONify jQuery插件。下面的示例是从GitHub README逐字获取的。全部归功于该插件的作者Kushal Pandya。
鉴于:
<form id="myform">
<label>Name:</label>
<input type="text" name="name"/>
<label>Email</label>
<input type="text" name="email"/>
<label>Password</label>
<input type="password" name="password"/>
</form>
运行:
$('#myform').jsonify();
产生:
{"name":"Joe User","email":"joe@example.com","password":"mypass"}
如果要使用此JSON对象执行jQuery POST:
$('#mybutton').click(function() {
$.post('/api/user', JSON.stringify($('#myform').jsonify()));
}