如何在 属性 中搜索 属性 对象?

How do you search object for a property within property?

我有这个对象:

key = {
    spawn:{type:1,img:app.assets.get('assets/spawn.svg')},
    wall:{type:2,img:app.assets.get('assets/wall.svg')},   
    grass:{type:3,img:app.assets.get('assets/grass.svg')},
    spike:{type:4,img:app.assets.get('assets/spike.svg')},
    ground:{type:5,img:app.assets.get('assets/ground.svg')}
  };

我有一个只有类型的数组,我需要向其中添加给定的图像,该数组看起来像这样:

[{type:1,image:null},{type:3,image:null},{type:2,image:null},{type:2,image:null},{type:5,image:null}]

基本上我想循环数组,找到关键对象中的类型并获取给定的图像并将其保存到数组中。

有什么简单的方法可以做到这一点吗?

var key = {
    spawn:{type:1,img:app.assets.get('assets/spawn.svg')},
    wall:{type:2,img:app.assets.get('assets/wall.svg')},   
    grass:{type:3,img:app.assets.get('assets/grass.svg')},
    spike:{type:4,img:app.assets.get('assets/spike.svg')},
    ground:{type:5,img:app.assets.get('assets/ground.svg')}
  };
var typesArray = [{type:1,image:null},{type:3,image:null},{type:2,image:null},{type:2,image:null},{type:5,image:null}];

for(var i = 0, j = typesArray.length; i < j; i++)
{
    typesArray[i].image = getKeyObjectFromType(typesArray[i].type).img;
}

function getKeyObjectFromType(type)
{
    for(var k in key)
    {
        if(key[k].type == type)
        {
            return key[k];
        }
    }
    return {};
}
for (var i = 0; i < typesArray.length; i++) {
    for (prop in key) {
        if (key[prop].type === typesArray[i].type) {
            typesArray[i].image = key[prop].img;
        }
    }
}

它循环遍历数组 ("typesArray"),并且对于每个数组项,它遍历键中的所有对象以寻找具有相同 "type" 的对象。当它找到它时,它会获取该关键对象的 "img" 并保存到数组中。

给定键对象和对象数组,如下所示:

var key = {
    spawn:{type:1,img:app.assets.get('assets/spawn.svg')},
    wall:{type:2,img:app.assets.get('assets/wall.svg')},   
    grass:{type:3,img:app.assets.get('assets/grass.svg')},
    spike:{type:4,img:app.assets.get('assets/spike.svg')},
    ground:{type:5,img:app.assets.get('assets/ground.svg')}
};
var arr = [{type:1,image:null},{type:3,image:null},{type:2,image:null},{type:2,image:null},{type:5,image:null}];

我们可以先在对象key中创建一个属性数组,使迭代更简单。 然后遍历数组 arr,在每个成员上,使用 some 循环检查哪个图像按其类型属于该成员(有些返回第一个 true 并结束循环) .

如果您希望循环没有副作用,您可以将 forEach 更改为 map(并将返回的新数组分配给 arr 或新变量) , 而不是改变原始数组。

var keyTypes = Object.keys(key);
arr.forEach(function (item) {
    keyTypes.some(function (keyType) {
        if (key[keyType].type === item.type) {
            item.image = key[keyType].img;
            return true;
        }
        return false;
    });
});

更明智的做法是更改图像类型的对象,以便您可以使用该类型作为访问对象 属性,或者为此创建另一个对象(如另一个答案中所指出的)。

我不确定这个解决方案是否现代,但它不使用任何循环或递归。

object = {
    spawn: {type:1, img:app.assets.get('assets/spawn.svg')},
    wall: {type:2, img:app.assets.get('assets/wall.svg')},   
    grass: {type:3, img:app.assets.get('assets/grass.svg')},
    spike: {type:4, img:app.assets.get('assets/spike.svg')},
    ground: {type:5, img:app.assets.get('assets/ground.svg')}
};
arr = [
    {type:1, image:null},
    {type:3, image:null},
    {type:2, image:null},
    {type:2, image:null},
    {type:5, image:null}
];

var typeImages = {};
Object.getOwnPropertyNames(object).forEach(function(value){
    typeImages[object[value].type] = object[value].img;
});
arr = arr.map(function(value){
    return {
        type: value.type,
        image: typeImages[value.type]
    };
});

使用 lodash (https://lodash.com/):

var key = {
    spawn:{type:1,img:app.assets.get('assets/spawn.svg')},
    wall:{type:2,img:app.assets.get('assets/wall.svg')},   
    grass:{type:3,img:app.assets.get('assets/grass.svg')},
    spike:{type:4,img:app.assets.get('assets/spike.svg')},
    ground:{type:5,img:app.assets.get('assets/ground.svg')}
};

var initialList = [{type:1,image:null},{type:3,image:null},{type:2,image:null},{type:2,image:null},{type:5,image:null}];

var updatedList = _.transform(initialList, function(result, item) {
    item.image = _.find(key, _.matchesProperty('type', item.type)).img;
    result.push(item);
});

这将遍历 initialList 中的每个项目,在 key 中找到与他们的 type 属性 匹配的对象并将其放入 image 属性.

最终结果将在updatedList

对我来说最突出的一件事是

...get the given image and save it into the array

我假设这意味着原始数组。我认为更好的方法是将适当的键和值映射到新数组,但对于这个示例,我假设这是一项要求。

试图使解决方案尽可能简洁并请求 lodash 解决方案:

_.each(key, function(prop){
    _.each(_.filter(types, { type: prop.type }), function(type) { type.image = prop.img });
});