[英]In JavaScript, how to conditionally add a member to an object?
我想创建一个 object 并有条件地添加一个成员。 简单的方法是:
var a = {};
if (someCondition)
a.b = 5;
现在,我想写一个更地道的代码。 我在尝试:
a = {
b: (someCondition? 5 : undefined)
};
但是现在, b
是a
的成员,其值为undefined
。 这不是想要的结果。
有方便的解决方案吗?
更新
我寻求一种解决方案,可以处理多个成员的一般情况。
a = {
b: (conditionB? 5 : undefined),
c: (conditionC? 5 : undefined),
d: (conditionD? 5 : undefined),
e: (conditionE? 5 : undefined),
f: (conditionF? 5 : undefined),
g: (conditionG? 5 : undefined),
};
我认为@InspiredJW 是用 ES5 做到的,正如@trincot 所指出的,使用 es6 是一种更好的方法。 但是我们可以通过使用扩展运算符和逻辑与短路评估来添加更多的糖:
const a = {
...(someCondition && {b: 5})
}
const obj = {
...(condition) && {someprop: propvalue},
...otherprops
}
现场演示:
const obj = { ...(true) && {someprop: 42}, ...(false) && {nonprop: "foo"}, ...({}) && {tricky: "hello"}, } console.log(obj);
在纯 Javascript 中,我想不出比您的第一个代码片段更惯用的东西了。
但是,如果使用 jQuery 库不是不可能的,那么$.extend()应该满足您的要求,因为正如文档所述:
未定义的属性不会被复制。
因此,您可以编写:
var a = $.extend({}, {
b: conditionB ? 5 : undefined,
c: conditionC ? 5 : undefined,
// and so on...
});
并获得您期望的结果(如果conditionB
为false
,则b
将不存在于a
中)。
我建议如下:
const a = {
...(someCondition? {b: 5}: {})
}
使用 EcmaScript2015,您可以使用Object.assign
:
Object.assign(a, conditionB ? { b: 1 } : null,
conditionC ? { c: 2 } : null,
conditionD ? { d: 3 } : null);
var a, conditionB, conditionC, conditionD; conditionC = true; a = {}; Object.assign(a, conditionB ? { b: 1 } : null, conditionC ? { c: 2 } : null, conditionD ? { d: 3 } : null); console.log(a);
一些备注:
Object.assign
就地修改了第一个参数,但它也返回了更新后的对象:因此您可以在更大的表达式中使用此方法,以进一步操作该对象。undefined
或{}
而不是null
,但结果相同。 您甚至可以提供0
代替,因为原始值被包装,并且Number
没有自己的可枚举属性。进一步考虑第二点,您可以将其缩短如下(正如@Jamie 指出的那样),因为虚假值没有自己的可枚举属性( false
、 0
、 NaN
、 null
、 undefined
、 ''
,除了document.all
):
Object.assign(a, conditionB && { b: 1 },
conditionC && { c: 2 },
conditionD && { d: 3 });
var a, conditionB, conditionC, conditionD; conditionC = "this is truthy"; conditionD = NaN; // falsy a = {}; Object.assign(a, conditionB && { b: 1 }, conditionC && { c: 2 }, conditionD && { d: 3 }); console.log(a);
有条件地向对象添加成员
const trueCondition = true;
const falseCondition = false;
const obj = {
...(trueCondition && { student: 10 }),
...(falseCondition && { teacher: 2 }),
};
// { student: 10 }
性能测试
经典方法
const a = {};
if (someCondition)
a.b = 5;
VS
传播算子方法
const a2 = {
...(someCondition && {b: 5})
}
结果:
经典方法要快得多,因此请考虑到语法糖化较慢。
testClassicConditionFulfilled(); // ~ 234.9 毫秒
testClassicConditionNotFulfilled(); // ~493.1ms
testSpreadOperatorConditionFulfilled(); // ~2649.4ms
testSpreadOperatorConditionNotFulfilled(); // ~2278.0ms
function testSpreadOperatorConditionFulfilled() { const value = 5; console.time('testSpreadOperatorConditionFulfilled'); for (let i = 0; i < 200000000; i++) { let a = { ...(value && {b: value}) }; } console.timeEnd('testSpreadOperatorConditionFulfilled'); } function testSpreadOperatorConditionNotFulfilled() { const value = undefined; console.time('testSpreadOperatorConditionNotFulfilled'); for (let i = 0; i < 200000000; i++) { let a = { ...(value && {b: value}) }; } console.timeEnd('testSpreadOperatorConditionNotFulfilled'); } function testClassicConditionFulfilled() { const value = 5; console.time('testClassicConditionFulfilled'); for (let i = 0; i < 200000000; i++) { let a = {}; if (value) ab = value; } console.timeEnd('testClassicConditionFulfilled'); } function testClassicConditionNotFulfilled() { const value = undefined; console.time('testClassicConditionNotFulfilled'); for (let i = 0; i < 200000000; i++) { let a = {}; if (value) ab = value; } console.timeEnd('testClassicConditionNotFulfilled'); } testClassicConditionFulfilled(); // ~ 234.9ms testClassicConditionNotFulfilled(); // ~493.1ms testSpreadOperatorConditionFulfilled(); // ~2649.4ms testSpreadOperatorConditionNotFulfilled(); // ~2278.0ms
更简化,
const a = {
...(condition && {b: 1}) // if condition is true 'b' will be added.
}
使用增强的对象属性怎么样,并且只在它是真实的情况下设置属性,例如:
[isConditionTrue() && 'propertyName']: 'propertyValue'
因此,如果不满足条件,则不会创建首选属性,因此您可以丢弃它。 请参阅: http ://es6-features.org/#ComputedPropertyNames
更新:最好遵循 Axel Rauschmayer 在他的博客文章中关于有条件地在对象文字和数组中添加条目的方法( http://2ality.com/2017/04/conditional-literal-entries.html ):
const arr = [
...(isConditionTrue() ? [{
key: 'value'
}] : [])
];
const obj = {
...(isConditionTrue() ? {key: 'value'} : {})
};
对我帮助很大。
更好的答案:
const a = {
...(someCondition ? {b: 5} : {})
}
简单的 ES6 解决方案
带有 (&) 的单一条件
const didIPassExam = true const study = { monday: 'writing', tuesday: 'reading', /* check conditionally and if true, then add wednesday to study */...(didIPassExam && {wednesday: 'sleep happily'}) } console.log(study)
具有 (? :) 的双重条件
const score = 110 //const score = 10 const storage = { a:10, b:20, ...(score > 100? {c: 30}: {d:40}) } console.log(storage)
解释
假设您有这样的storage
object
const storage = {
a : 10,
b : 20,
}
并且您想根据score
有条件地为此添加道具
const score = 90
如果score
大于100
,您现在想将c:30
添加到storage
中。
如果分数小于100
,那么您要将d:40
添加到storage
中。 你可以这样做
const score = 110
const storage = {
a:10,
b:20,
...(score > 100 ? {c: 30} : {d:40})
}
上面的代码给出了storage
作为
{
a: 10,
b: 20,
c: 30
}
如果score = 90
然后你得到storage
作为
{
a: 10,
b: 20,
d: 40
}
这可能是 ES6 中最短的解决方案
console.log({
...true && {foo: 'bar'}
})
// Output: {foo:'bar'}
console.log({
...false && {foo: 'bar'}
})
// Output: {}
我会这样做
var a = someCondition ? { b: 5 } : {};
您可以无条件添加所有未定义的值,然后使用JSON.stringify
将它们全部删除:
const person = {
name: undefined,
age: 22,
height: null
}
const cleaned = JSON.parse(JSON.stringify(person));
// Contents of cleaned:
// cleaned = {
// age: 22,
// height: null
// }
如果目标是让对象看起来是独立的并且在一组大括号内,你可以试试这个:
var a = new function () {
if (conditionB)
this.b = 5;
if (conditionC)
this.c = 5;
if (conditionD)
this.d = 5;
};
这早就得到了回答,但是看着其他想法,我想出了一些有趣的衍生物:
使用匿名构造函数创建您的对象,并始终将未定义的成员分配给您在最后删除的同一虚拟成员。 这将为每个成员提供一行(我希望不是太复杂)+ 1 个附加行。
var a = new function() {
this.AlwaysPresent = 1;
this[conditionA ? "a" : "undef"] = valueA;
this[conditionB ? "b" : "undef"] = valueB;
this[conditionC ? "c" : "undef"] = valueC;
this[conditionD ? "d" : "undef"] = valueD;
...
delete this.undef;
};
如果你希望做这个服务器端(没有 jquery),你可以使用 lodash 4.3.0:
a = _.pickBy({ b: (someCondition? 5 : undefined) }, _.negate(_.isUndefined));
这可以使用 lodash 3.10.1
a = _.pick({ b: (someCondition? 5 : undefined) }, _.negate(_.isUndefined));
var a = {
...(condition ? {b: 1} : '') // if condition is true 'b' will be added.
}
我希望这是根据条件添加条目的有效方法。 有关如何有条件地在对象文字中添加条目的更多信息。
我用另一种选择做了一个小基准测试。 我喜欢从一些物体上去除“自重”。 通常是虚假值。
以下是本benny
的结果:
const clean = o => {
for (const prop in o) if (!o) delete o[prop];
}
clean({ value });
let a = {
...(value && {b: value})
};
let a = {};
if (value) {
a.b = value;
}
clean : 84 918 483 ops/s, ±1.16% | 51.58% slower
spread : 20 188 291 ops/s, ±0.92% | slowest, 88.49% slower
if : 175 368 197 ops/s, ±0.50% | fastest
使用 lodash 库,您可以使用_.omitBy
var a = _.omitBy({
b: conditionB ? 4 : undefined,
c: conditionC ? 5 : undefined,
}, _.IsUndefined)
当您有可选的请求时,这很方便
var a = _.omitBy({
b: req.body.optionalA, //if undefined, will be removed
c: req.body.optionalB,
}, _.IsUndefined)
这是我能想到的最简洁的解决方案:
var a = {};
conditionB && a.b = 5;
conditionC && a.c = 5;
conditionD && a.d = 5;
// ...
我更喜欢,使用它的代码,你可以运行这个代码
const three = {
three: 3
}
// you can active this code, if you use object `three is null`
//const three = {}
const number = {
one: 1,
two: 2,
...(!!three && three),
four: 4
}
console.log(number);
const isAdult = true;
const obj = {
...(isAdult ? { age: 18 }: { age: 17}),
};
//>> { student: 18 }
我认为您第一种有条件地添加成员的方法非常好。 我真的不同意不希望 a 的成员b
a
值为undefined
。 使用带有in
运算符的for
循环添加undefined
检查非常简单。 但无论如何,您可以轻松编写一个函数来过滤掉undefined
的成员。
var filterUndefined = function(obj) {
var ret = {};
for (var key in obj) {
var value = obj[key];
if (obj.hasOwnProperty(key) && value !== undefined) {
ret[key] = value;
}
}
return ret;
};
var a = filterUndefined({
b: (conditionB? 5 : undefined),
c: (conditionC? 5 : undefined),
d: (conditionD? 5 : undefined),
e: (conditionE? 5 : undefined),
f: (conditionF? 5 : undefined),
g: (conditionG? 5 : undefined),
});
您还可以使用delete
运算符来编辑对象。
下面的代码片段应该可以工作。
const a = {}
const conditionB = true;
const conditionC = true;
const conditionD = true;
const conditionE = true;
const b = {
...(conditionB && { b : 5}),
...(conditionC && { c : 5}),
...(conditionD && { d : 5}),
...(conditionE && { e : 5}),
};
console.log(b);
我希望这有助于解决您的问题
<body> <h1>GeeksforGeeks</h1> <p id="geeks"></p> <:-- Script to check array include object or not --> <script> var obj = {"geeks1",10: "geeks2",12} var arr = ["geeks1", "geeks2", "geeks3"; obj]. if(arr.filter(value=> value==obj).length > 0) document;write("true"). else document;write("false"); </script> </body>
使用 lodash 库,您可以使用_.merge
var a = _.merge({}, {
b: conditionB ? 4 : undefined,
c: conditionC ? 5 : undefined,
})
false
且 conditionC 为true
,则a = { c: 5 }
true
,则a = { b: 4, c: 5 }
false
,则a = {}
包装成一个对象
像这样的东西有点干净
const obj = {
X: 'dataX',
Y: 'dataY',
//...
}
const list = {
A: true && 'dataA',
B: false && 'dataB',
C: 'A' != 'B' && 'dataC',
D: 2000 < 100 && 'dataD',
// E: conditionE && 'dataE',
// F: conditionF && 'dataF',
//...
}
Object.keys(list).map(prop => list[prop] ? obj[prop] = list[prop] : null)
包装成一个数组
或者,如果您想使用 Jamie Hill 的方法并且有很长的条件列表,那么您必须多次编写...
语法。 为了使其更简洁,您可以将它们包装到一个数组中,然后使用reduce()
将它们作为单个对象返回。
const obj = {
X: 'dataX',
Y: 'dataY',
//...
...[
true && { A: 'dataA'},
false && { B: 'dataB'},
'A' != 'B' && { C: 'dataC'},
2000 < 100 && { D: 'dataD'},
// conditionE && { E: 'dataE'},
// conditionF && { F: 'dataF'},
//...
].reduce(( v1, v2 ) => ({ ...v1, ...v2 }))
}
或使用map()
函数
const obj = {
X: 'dataX',
Y: 'dataY',
//...
}
const array = [
true && { A: 'dataA'},
false && { B: 'dataB'},
'A' != 'B' && { C: 'dataC'},
2000 < 100 && { D: 'dataD'},
// conditionE && { E: 'dataE'},
// conditionF && { F: 'dataF'},
//...
].map(val => Object.assign(obj, val))
通过let
定义一个 var 并分配新属性
let msg = {
to: "hito@email.com",
from: "hifrom@email.com",
subject: "Contact form",
};
if (file_uploaded_in_form) { // the condition goes here
msg.attachments = [ // here 'attachments' is the new property added to msg Javascript object
{
content: "attachment",
filename: "filename",
type: "mime_type",
disposition: "attachment",
},
];
}
现在msg
变成
{
to: "hito@email.com",
from: "hifrom@email.com",
subject: "Contact form",
attachments: [
{
content: "attachment",
filename: "filename",
type: "mime_type",
disposition: "attachment",
},
]
}
在我看来,这是非常简单易行的解决方案。
为了完整起见,如果要添加其他描述符,可以使用Object.defineProperty()
。 请注意,我特意添加了enumerable: true
,否则该属性不会出现在console.log()
中。 这种方法的优点是,如果要添加多个新属性,还可以使用Object.defineProperties()
(但是,这样每个属性都将依赖于相同的条件......)
const select = document.getElementById("condition"); const output = document.getElementById("output"); let a = {}; let b = {}; select.onchange = (e) => { const condition = e.target.value === "true"; condition ? Object.defineProperty(a, "b", { value: 5, enumerable: true, }) : (a = {}); condition ? Object.defineProperties(b, { c: { value: 5, enumerable: true, }, d: { value: 6, enumerable: true, }, e: { value: 7, enumerable: true, }, }) : (b = {}); outputSingle.innerText = JSON.stringify(a); outputMultiple.innerText = JSON.stringify(b); };
Condition: <select id="condition"> <option value="false">false</option> <option value="true">true</option> </select> <br/> <br/> Single Property: <pre id="outputSingle">{}</pre><br/> Multiple Properties: <pre id="outputMultiple">{}</pre>
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.