提问人:Rafay 提问时间:11/29/2011 最后编辑:lolbasRafay 更新时间:4/8/2023 访问量:1617046
循环访问对象属性
Iterate through object properties
问:
var obj = {
name: "Simon",
age: "20",
clothing: {
style: "simple",
hipster: false
}
}
for(var propt in obj){
console.log(propt + ': ' + obj[propt]);
}
变量如何表示对象的属性?它不是内置方法或属性。为什么它会出现对象中的每个属性?propt
答:
它是(MDN,ECMAScript规范)。for...in statement
您可以将其理解为“对于对象中的每个属性,依次将每个属性分配给 PROPT 变量”。obj
评论
for-in
语句本身代表语法生成:,in
for
for ( LeftHandSideExpression in Expression )
for ( var VariableDeclarationNoIn in Expression )
这只是一个循环。查看 Mozilla 上的文档。for...in
评论
您的循环正在遍历对象的所有属性。 在 for 循环的第一行中定义。它是一个字符串,是对象属性的名称。在循环的第一次迭代中,将是“name”。for
obj
propt
obj
propt
JavaScript 中的对象是属性的集合,因此可以在 for 每个语句中循环。
应将其视为键值集合。obj
评论
循环访问属性需要以下额外检查:hasOwnProperty
for (var prop in obj) {
if (Object.prototype.hasOwnProperty.call(obj, prop)) {
// do stuff
}
}
这是必要的,因为对象的原型包含对象的其他属性,这些属性在技术上是对象的一部分。这些附加属性继承自基对象类,但仍是 的属性。obj
hasOwnProperty
只是检查这是否是特定于此类的属性,而不是从基类继承的属性。
也可以通过对象本身进行调用:hasOwnProperty
if (obj.hasOwnProperty(prop)) {
// do stuff
}
但是,如果对象具有同名的不相关字段,则此操作将失败:
var obj = { foo: 42, hasOwnProperty: 'lol' };
obj.hasOwnProperty('foo'); // TypeError: hasOwnProperty is not a function
这就是为什么调用它更安全的原因:Object.prototype
var obj = { foo: 42, hasOwnProperty: 'lol' };
Object.prototype.hasOwnProperty.call(obj, 'foo'); // true
评论
Object.prototype
I couldn't get my head around "looping through object properties"
hasOwnProperty
Object.Prototype
property
propertyName
if
从 JavaScript 1.8.5 开始,您可以使用它来获取在对象本身上定义的属性数组(返回 true 的属性数组)。Object.keys(obj)
obj.hasOwnProperty(key)
Object.keys(obj).forEach(function(key,index) {
// key: the name of the object key
// index: the ordinal position of the key within the object
});
这比使用 for-in 循环更好(也更易读)。
以下浏览器支持它:
- Firefox (Gecko):4 (2.0)
- 铬:5
- IE浏览器:9
有关更多信息,请参阅Mozilla开发人员网络Object.keys()的参考。
评论
Object.keys(myObject).forEach(function(key,index) { //key = the name of the object key //index = the ordinal position of the key within the object });
for candidate in candidateStatus
为什么。。in 循环的作用是创建一个新变量(var someVariable),然后将给定对象的每个属性逐个存储在这个新变量(someVariable)中。因此,如果使用块 {},则可以迭代。请看以下示例。
var obj = {
name:'raman',
hobby:'coding',
planet:'earth'
};
for(var someVariable in obj) {
//do nothing..
}
console.log(someVariable); // outputs planet
评论
obj[someVariable]
为了...in 循环表示对象中的每个属性,因为它就像一个 for 循环。您在 for...通过执行以下操作在循环中:
for(var propt in obj){
alert(propt + ': ' + obj[propt]);
}
一个为...in 循环循环访问对象的可枚举属性。无论您定义哪个变量,或放入 for...在循环中,每次转到它迭代的下一个属性时都会更改。for...in 循环遍历键,但它的值是键的值。例如:
for(var propt in obj) {
console.log(propt);//logs name
console.log(obj[propt]);//logs "Simon"
}
您可以看到变量与变量值的不同之处。相比之下,一个 for...of 循环则相反。
我希望这会有所帮助。
我想补充上面的答案,因为你可能对 Javascript 有不同的意图。JSON 对象和 Javascript 对象是不同的东西,您可能希望使用上面建议的解决方案遍历 JSON 对象的属性,然后会感到惊讶。
假设您有一个 JSON 对象,如下所示:
var example = {
"prop1": "value1",
"prop2": [ "value2_0", "value2_1"],
"prop3": {
"prop3_1": "value3_1"
}
}
迭代其“属性”的错误方法:
function recursivelyIterateProperties(jsonObject) {
for (var prop in Object.keys(example)) {
console.log(prop);
recursivelyIterateProperties(jsonObject[prop]);
}
}
您可能会惊讶地看到控制台日志记录 、 等。 遍历 和 和 的属性。这些对象是序列,序列的索引是该对象在 Javascript 中的属性。0
1
prop1
prop2
prop3_1
递归遍历 JSON 对象属性的更好方法是首先检查该对象是否为序列:
function recursivelyIterateProperties(jsonObject) {
for (var prop in Object.keys(example)) {
console.log(prop);
if (!(typeof(jsonObject[prop]) === 'string')
&& !(jsonObject[prop] instanceof Array)) {
recursivelyIterateProperties(jsonObject[prop]);
}
}
}
同时添加递归方式:
function iterate(obj) {
// watch for objects we've already iterated so we won't end in endless cycle
// for cases like var foo = {}; foo.bar = foo; iterate(foo);
var walked = [];
var stack = [{obj: obj, stack: ''}];
while(stack.length > 0)
{
var item = stack.pop();
var obj = item.obj;
for (var property in obj) {
if (obj.hasOwnProperty(property)) {
if (typeof obj[property] == "object") {
// check if we haven't iterated through the reference yet
var alreadyFound = false;
for(var i = 0; i < walked.length; i++)
{
if (walked[i] === obj[property])
{
alreadyFound = true;
break;
}
}
// new object reference
if (!alreadyFound)
{
walked.push(obj[property]);
stack.push({obj: obj[property], stack: item.stack + '.' + property});
}
}
else
{
console.log(item.stack + '.' + property + "=" + obj[property]);
}
}
}
}
}
用法:
iterate({ foo: "foo", bar: { foo: "foo"} });
评论
在这里,我将迭代每个节点并创建有意义的节点名称。如果您注意到,instanceOf Array 和 instanceOf Object 几乎做同样的事情(在我的应用程序中,我给出了不同的逻辑)
function iterate(obj,parent_node) {
parent_node = parent_node || '';
for (var property in obj) {
if (obj.hasOwnProperty(property)) {
var node = parent_node + "/" + property;
if(obj[property] instanceof Array) {
//console.log('array: ' + node + ":" + obj[property]);
iterate(obj[property],node)
} else if(obj[property] instanceof Object){
//console.log('Object: ' + node + ":" + obj[property]);
iterate(obj[property],node)
}
else {
console.log(node + ":" + obj[property]);
}
}
}
}
注意 - 我受到 Ondrej Svejdar 回答的启发。但此解决方案具有更好的性能和更少的歧义
jQuery现在允许您执行此操作:
$.each( obj, function( key, value ) {
alert( key + ": " + value );
});
评论
$.each({foo:1, length:0, bar:2}, function(k,v){console.log(k,v)})
$.each 不适合对象。如果一个对象恰好有一个 length 属性,并且它的值恰好为零,则整个对象将被视为一个空数组。
基本上,您希望遍历对象中的每个属性。
var Dictionary = {
If: {
you: {
can: '',
make: ''
},
sense: ''
},
of: {
the: {
sentence: {
it: '',
worked: ''
}
}
}
};
function Iterate(obj) {
for (prop in obj) {
if (obj.hasOwnProperty(prop) && isNaN(prop)) {
console.log(prop + ': ' + obj[prop]);
Iterate(obj[prop]);
}
}
}
Iterate(Dictionary);
现在,只需添加 Symbol.iterator 方法,即可将标准 JS 对象转换为可迭代对象。然后,您可以使用循环并直接访问其值,甚至可以在对象上使用扩展运算符。凉。让我们看看如何做到这一点:for of
var o = {a:1,b:2,c:3},
a = [];
o[Symbol.iterator] = function*(){
var ok = Object.keys(this);
i = 0;
while (i < ok.length) yield this[ok[i++]];
};
for (var value of o) console.log(value);
// or you can even do like
a = [...o];
console.log(a);
评论
function*
在 ES 的最新实现中,您可以使用 Object.entries
:
for (const [key, value] of Object.entries(obj)) { }
或
Object.entries(obj).forEach(([key, value]) => ...)
如果只想遍历值,请使用 Object.values:
for (const value of Object.values(obj)) { }
或
Object.values(obj).forEach(value => ...)
评论
for (const [key, value] of Object.entries(obj)) { }
您可以使用 Lodash。文档
var obj = {a: 1, b: 2, c: 3};
_.keys(obj).forEach(function (key) {
...
});
评论
女孩和男孩,我们在 2019 年,我们没有那么多时间打字......因此,让我们做这个很酷的新花哨的 ECMAScript 2016:
Object.keys(obj).forEach(e => console.log(`key=${e} value=${obj[e]}`));
评论
obj=window.performance.memory
for in
var obj = window.performance.memory; for( key in obj ) console.log( 'key=' + key + ' val=' + obj[key] );
window.performance.memory
仅受 chrome 支持,并返回一个空数组。这与.Object.keys(obj)
.map
e
(
(key)
=>
(value)
)
{
key
=>
value
}
let obj = {"a": 3, "b": 2, "6": "a"}
Object.keys(obj).forEach((item) => {console.log("item", obj[item])})
// a
// 3
// 2
评论
forEach
map
map
如果运行 Node,我建议:
Object.keys(obj).forEach((key, index) => {
console.log(key);
});
为了进一步细化已接受的答案,值得注意的是,如果使用 then 实例化对象,将触发 TypeError。因此,为了安全起见,您需要从原型中调用它,如下所示:var object = Object.create(null)
object.hasOwnProperty(property)
for (var property in object) {
if (Object.prototype.hasOwnProperty.call(object, property)) {
// do stuff
}
}
如果您的环境支持 ES2017,那么我推荐 Object.entries:
Object.entries(obj).forEach(([key, value]) => {
console.log(`${key} ${value}`);
});
如 Mozillas Object.entries() 文档所示:
Object.entries() 方法返回给定对象自己的数组 可枚举属性 [key, value] 对,顺序相同 由一个提供...in 循环(区别在于 for-in 循环 也枚举原型链中的属性)。
基本上,使用 Object.entrys,我们可以放弃以下额外的步骤,这是较旧的 for...循环中:
// This step is not necessary with Object.entries
if (object.hasOwnProperty(property)) {
// do stuff
}
上面的答案有点烦人,因为它们没有解释你在确保它是一个对象后在 for 循环中做了什么:你不直接访问它!实际上,您只收到了应用于 OBJ 所需的 KEY:
var obj = {
a: "foo",
b: "bar",
c: "foobar"
};
// We need to iterate the string keys (not the objects)
for(var someKey in obj)
{
// We check if this key exists in the obj
if (obj.hasOwnProperty(someKey))
{
// someKey is only the KEY (string)! Use it to get the obj:
var myActualPropFromObj = obj[someKey]; // Since dynamic, use [] since the key isn't literally named "someKey"
// NOW you can treat it like an obj
var shouldBeBar = myActualPropFromObj.b;
}
}
这一切都是 ECMA5 安全的。甚至可以在蹩脚的JS版本中工作,如Rhino;)
评论
shouldBeBar
对于所有三个迭代,均未定义。
undefined
是不可能的,因为它在你面前被清楚地定义:)这扣除了 2 种可能性:(1) 您是否在 if() 块之外检查 shouldBeBar?那么是的,它将是未定义的(超出范围)。或者 (2) 你打错了 var 名称吗?
多米尼克的回答很完美,我只是更喜欢这样做,因为它读起来更干净:
for (var property in obj) {
if (!obj.hasOwnProperty(property)) continue;
// Do stuff...
}
评论
Object
object
Object.keys(obj).forEach(key =>
console.log(`key=${key} value=${obj[key]}`)
);
添加 ES2015 对属性的用法,并通过迭代器迭代属性。Reflect.ownKeys(obj)
例如:
let obj = { a: 'Carrot', b: 'Potato', Car: { doors: 4 } };
可以通过以下方式进行迭代
// logs each key
Reflect.ownKeys(obj).forEach(key => console.log(key));
如果要直接遍历对象键的值,可以定义一个 ,就像 JavaScipts 的字符串、数组、类型化数组、Map 和 Set 的默认迭代器一样。iterator
JS 将尝试通过默认迭代器属性进行迭代,该属性必须定义为 .Symbol.iterator
如果您希望能够遍历所有对象,可以将其添加为 Object 的原型:
Object.prototype[Symbol.iterator] = function*() {
for(p of Reflect.ownKeys(this)){ yield this[p]; }
}
这将使您能够循环访问对象的值,并带有...of 循环,例如:
for(val of obj) { console.log('Value is:' + val ) }
注意:在撰写此答案时(2018 年 6 月),除 IE 外,所有其他浏览器都支持生成器并通过for...of
Symbol.iterator
评论
虽然最受好评的答案是正确的,但这里有一个替代用例,即如果你正在迭代一个对象并最终想要创建一个数组。使用代替.map
forEach
const newObj = Object.keys(obj).map(el => {
//ell will hold keys
// Getting the value of the keys should be as simple as obj[el]
})
if (typeof obj === 'object' && obj !== null) {
Object.keys(obj).forEach(key => {
console.log("\n" + key + ": " + obj[key]);
});
}
// *** Explanation line by line ***
// Explaining the bellow line
// It checks if obj is neither null nor undefined, which means it's safe to get its keys.
// Otherwise it will give you a "TypeError: Cannot convert undefined or null to object" if obj is null or undefined when it tries to get its keys in the next line of code.
// NOTE 1: You can use Object.hasOwnProperty() instead of Object.keys(obj).length
// NOTE 2: No need to check if obj is an array because it will work just fine.
// NOTE 3: No need to check if obj is a string because it will not pass the 'if typeof obj is Object' statement.
// NOTE 4: No need to check if Obj is undefined because it will not pass the 'if type obj is Object' statement either.
if (typeof obj === 'object' && obj !== null) {
// Explaining the bellow line
// Just like in the previous line, this returns an array with
// all keys in obj (because if code execution got here, it means
// obj has keys.)
// Then just invoke built-in javascript forEach() to loop
// over each key in returned array and calls a call back function
// on each array element (key), using ES6 arrow function (=>)
// Or you can just use a normal function ((key) { blah blah }).
Object.keys(obj).forEach(key => {
// The bellow line prints out all keys with their
// respective value in obj.
// key comes from the returned array in Object.keys(obj)
// obj[key] returns the value of key in obj
console.log("\n" + key + ": " + obj[key]);
});
}
评论
forEach
Object.keys(obj).forEach(e => console.log(`key=${e} value=${obj[e]}`));
检查类型
您可以通过以下方式检查 propt 如何表示对象 propertis
typeof propt
发现它只是一个字符串(属性名称)。由于 for-in
js“内置”循环的工作方式,它提出了对象中的每个属性。
var obj = {
name: "Simon",
age: "20",
clothing: {
style: "simple",
hipster: false
}
}
for(var propt in obj){
console.log(typeof propt, propt + ': ' + obj[propt]);
}
您可以使用 and 循环访问对象的嵌套属性。for...in
forEach
为。。。在:
for (const key in info) {
console.log(info[key]);
}
forEach:
Object.keys(info).forEach(function(prop) {
console.log(info[prop]);
// cities: Array[3], continent: "North America", images: Array[3], name: "Canada"
// "prop" is the property name
// "data[prop]" is the property value
});
如果您只想迭代以映射属性值,那么 lodash 具有_.mapValues
const obj = {
a: 2,
b: 3
}
const res = _.mapValues(obj, v => v * 2)
console.log(res)
<script src="https://cdn.jsdelivr.net/npm/[email protected]/lodash.min.js"></script>
上一个:如何以块形式迭代列表
评论
if (typeof(obj[propt]) === 'object') {
/* 再做一次 */}