开发者问题收集

如何正确搜索 javascript 对象内部

2016-08-16
1256

我有一个简单的 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 个函数

  1. 对于用户传递给脚本的每个类别(本例中为前面、其他),检查任何子类别中是否至少有一个值是 true ,如果是,则停止执行循环以避免无用的性能使用。
  2. 循环遍历所有子类别(无论父类别是什么),并检查至少一个值是否为 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
2个回答

来吧,试试这个:

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'));
Andy-Delosdos
2016-08-16
  • 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

Megajin
2016-08-16