如何正确搜索 javascript 对象内部
我有一个简单的 CLI 应用程序,用户可以在其中指定将导入到其项目的所有单个对象的类别(类似于 lodash 自定义构建)
为了避免重复数据,我创建了一个对象结构,如下所示:
{
"front": {
"subCategories": [{
"name": "front1",
"value": false
}, {
"name": "front2",
"value": true
}]
},
"other": {
"subCategories": [{
"name": "other1",
"value": false
}, {
"name": "other2",
"value": false
}]
}
}
并从用户输入中获取
-
类别名称,例如字符串
front
(但它也可以列出所有类别) -
或者对象名称数组,例如
['front1', 'other2']
根据此输入,我必须确定是否至少有一个值为
true
。需要注意的是,问题不是要知道用户发送的是类别还是单个对象,我从输入解析器中知道这一点 :)
所以我需要编写 2 个函数
-
对于用户传递给脚本的每个类别(本例中为前面、其他),检查任何子类别中是否至少有一个值是
true
,如果是,则停止执行循环以避免无用的性能使用。 -
循环遍历所有子类别(无论父类别是什么),并检查至少一个值是否为
true
,其中名称来自输入的对象名称数组,例如['front1', 'other2']
我尝试过:
我可以想象,如果我将
objValue=false;
设置为默认值,然后将 subCategories 对象传递给类似于下面循环内的代码,可能会起作用:
var objValue = _.result(_.find(myObj, function(obj) {
return obj.name === name;
}), 'value');
在我调用它之前,我只是询问
if(objValue) {//code here
但是这有点像最后一步,我不知何故在开始时就迷路了。
如果有人可以指导我如何正确操作:
-
查找对象中是否有任何值将值设置为
true
-
在所有 subCategories(可能远不止 2 个)对象中查找是否有一个对象的值设置为
true
来吧,试试这个:
var data = {
"front": {
"subCategories": [{
"name": "front1",
"value": false
}, {
"name": "front2",
"value": false
}]
},
"other": {
"subCategories": [{
"name": "other1",
"value": false
}, {
"name": "other2",
"value": true
}]
}
}
function searchSubCategories(obj, subCats)
{
var ret = false;
for (var property in obj)
{
for (var i=0; i <obj[property].subCategories.length; i++){
for (var x =0; x < subCats.length; x++)
{
ret = (obj[property].subCategories[i].name === subCats[x] && obj[property].subCategories[i].value === true);
if (ret)
break;
}
if (ret)
break;
}
if (ret)
break;
}
return ret;
}
function searchCategory(obj, category){
var ret = false;
for (var i=0; i <obj[category].subCategories.length; i++)
{
ret = obj[category].subCategories[i].value;
if (ret)
break;
}
return ret;
}
console.log(searchSubCategories(data, ['front1', 'other2']));
console.log(searchCategory(data, 'other'));
console.log(searchCategory(data, 'front'));
- find if any value in object have value set to true
- find in all subCategories(it can be a lot more then just 2) object if there is an object with value set to true
以下步骤应可帮助您了解如何开始所需操作:
1. JSON 中存储了多少个类别?
var myCategories = {
"front": {
"subCategorieA": [{
"name": "front1",
"value": false
}, {
"name": "front2",
"value": true
}],
"subCategorieB": [{
"name": "front1",
"value": false
}, {
"name": "front2",
"value": true
}]
},
"other": {
"subCategories": [{
"name": "other1",
"value": false
}, {
"name": "other2",
"value": false
}]
} };
我们有 front 和其他,总共 2 个。还有一些子类别。
在此示例中,我添加了 subCategoriesB,因为您提到可能还有很多。
2. 遍历查找类别
//setup a for loop to get through the objects
for(let key in myCategories){
/**
* do stuff
*/
}
请注意,在当前对象结构下,无法像数组一样遍历对象。例如,您不能执行以下操作:
myCategories[0]
// will throw error. An object is not an array!
3.现在我们可以遍历我们的对象,我们需要做的就是将所有对象打包在一起并获取子类别
我们创建一个将返回所需值的函数:
// To keep it simple a for loop which looks for Keys in Object
// Example {key: value} => {front: subCategorieA, subCategorieB}
for(let key in myCategories){
// Once we know the key, we can get the Value which is again an Object
// In our case subCategorieA and subCategorieB for Key front
getValuesFromSubcategories(myCategories[key]);
};
// This function simply returns the name of any value which is true
function getValuesFromSubcategories(mySubCategories){
//to get Indexes for the arrays we have stored in our objects
let counter = 0;
// looks familiar right?
for(let key in mySubCategories){
if(mySubCategories[key][counter].value === true){
//logs any Subcategorie name
console.log(mySubCategories[key][counter].name);
/**
* Do other needed stuff
*/
}
//increment correctly
counter++;
};
}
您看到我的评论“看起来很熟悉吧”?这就是您需要考虑递归的地方。但是既然您告诉我您是 Javascript 的初级程序员,我就不想再让您感到困惑了。请注意,您可以将所有这些打包成一个函数,该函数会在某个时候调用自身以进入对象的最深层。
但是我提供的最后一部分代码将记录任何为真的值的名称。您可以从这里根据需要对其进行自定义。
希望这能让您的问题变得清晰。 如果您有任何其他问题,请发表评论 =)。
问候,Megajin