Lodash:从多个构造单个对象 - 合并/覆盖属性

时间:2014-01-14 17:42:38

标签: javascript lodash

  

注意:我在lodash下提出了这个问题,因为我很确定它可以帮助我很好地解决这个问题,但是现在还没有把它放在手上

我有一个描述不同用户角色及其权限的对象;

我会像“这样”定义10-15个角色(这不反映应用程序代码,但问题本身):

    var role1 = {
    views: {
        v1: {access: true},
        v2: {access: false},
        v#: {access: false}
    }
}

var role2 = {
    views: {
        v1: {access: false},
        v2: {access: true},
        v3: {access: true},
    }
}

连接的用户将拥有多个角色;在该示例中,它可以是['role1', 'role2'],并且在此之外我需要构造单个permissions对象,该对象将是在所有用户角色中定义的所有道具的组合。

它基本上是基于白名单的,其中所有“true”属性都应该覆盖任何被定义为false的属性。因此,预期结果应为:

permissions = {
    views: {
        v1: {access: true},
        v2: {access: true},
        v2: {access: true}
    }
}

我不太确定如何在不依赖疯狂的嵌套循环的情况下解决这个问题

这是JSBin的起点:http://jsbin.com/usaQejOJ/1/edit?js,console

感谢您的帮助!

1 个答案:

答案 0 :(得分:41)

Lodash有一些方法可以帮助优雅地解决这个问题。

首先,merge方法。它需要多个源对象,并以递归方式将它们的属性合并到一个目标对象中。如果两个对象具有相同的属性名称,则后一个值将覆盖前一个。

这几乎是我们想要的;它会将您的角色对象合并为一个对象。我们不想要的是覆盖行为;我们希望true值始终覆盖false个值。幸运的是,你可以传入一个自定义合并函数,当两个对象具有相同的密钥时,lodash的merge将使用它来计算合并的值。

我们会编写一个自定义函数,将您的项目逻辑OR放在一起(而不是允许以后的false值覆盖true。),这样,如果任何一个值为{{} 1}},生成的合并值为true

因为我们的对象是嵌套的,所以我们需要确保我们的自定义合并函数只在比较布尔值时才执行此true。在比较对象时,我们希望对其属性进行正常合并(再次使用我们的自定义函数进行合并)。它看起来像这样:

OR

Lodash以另一种方式提供帮助:您可能会从文档中注意到function do_merge(roles) { // Custom merge function ORs together non-object values, recursively // calls itself on Objects. var merger = function (a, b) { if (_.isObject(a)) { return _.merge({}, a, b, merger); } else { return a || b; } }; // Allow roles to be passed to _.merge as an array of arbitrary length var args = _.flatten([{}, roles, merger]); return _.merge.apply(_, args); } do_merge([role1, role2, role3]); 不接受要合并的对象数组;你必须作为参数传递它们。但在我们的例子中,数组非常方便。

为了解决这个问题,我们将使用JavaScript的apply方法,它允许您通过将其参数作为数组传递来调用方法,以及Lodash方便的_.merge方法,该方法采用一个数组可能包含嵌套数组 - 如flatten - 并将其展平为[1, [2, 3], [4, 5]]

因此,我们会在[1, 2, 3, 4, 5] ed数组中收集我们需要的所有参数,然后flatten将它们收集到apply

如果你的对象非常深入地嵌套,那么你可能会像这样递归地调用merge的堆栈,但对于你的对象,这应该可以正常工作。