Javascript属性检查器功能(使用闭包)

时间:2018-12-06 22:24:40

标签: javascript closures

我正在尝试编写一个可用于检查对象属性的函数。

例如,对于给定的对象:

var obj = {
    s : 'a',
    n : 1,
    d : new Date(),
    f : function(){},
    a : [],
    o : {
        a : {
            b: 'b'
        }
    }
};

我可以使用检查器功能访问属性:

var ins = inspector(obj);
ins('s') // -> a
ins('n') // -> 1
ins('d') // -> 2018-12-06T11:51:26.244Z
ins('f') // -> [Function]
ins('a') // -> []
ins('o') // -> { a: { b: 'b' } }
ins('o.a') // -> { b: 'b' }
ins('o.a.b') // -> b
ins('o.p') // -> undefined
ins('o.p.p') // -> undefined
ins('p') // -> undefined
ins('') // -> undefined

我的第一种方法:

function inspector(obj){
    return function fun (prop){
        console.log(obj[prop])
    }
}

但是,在某些情况下它不能正常工作:

ins('o.a') // -> { b: 'b' }
ins('o.a.b') // -> b
ins('o.p') // -> undefined
ins('o.p.p') // -> undefined

如何改写我的函数以适应这些情况?

1 个答案:

答案 0 :(得分:5)

您可以使用reduce来获取值。首先,您需要按.分割路径:

function inspector(obj){
    return function fun (prop){
        return prop.split(".").reduce(function(o, p) {  // for each part p in prop.split(".")
            return o && o[p];                           // if o is an object truthy return o[p]
        }, obj);                                        // pass obj as the initial o
    }
}

测试o &&非常基础。如果仅在o是对象的情况下要对o下标,则将该行替换为:

return (o && typeof o === "object")? o[p]: undefined;

如果o[p]是对象,则返回o,否则返回undefined

示例:

function inspector(obj){
    return function fun (prop){
        return prop.split(".").reduce(function(o, p) {
            return (o && typeof o === "object")? o[p]: undefined;
        }, obj);
    }
}

var obj = {
    s : 'a',
    n : 1,
    d : new Date(),
    f : function(){},
    a : [],
    o : {
        a : {
            b: 'b'
        }
    }
};

var ins = inspector(obj);

console.log('s:', ins('s'));          // -> a
console.log('n:', ins('n'));          // -> 1
console.log('d:', ins('d'));          // -> 2018-12-06T11:51:26.244Z
console.log('f:', ins('f'));          // -> [Function]
console.log('a:', ins('a'));          // -> []
console.log('o:', ins('o'));          // -> { a: { b: 'b' } }
console.log('o.a:', ins('o.a'));      // -> { b: 'b' }
console.log('o.a.b:', ins('o.a.b'));  // -> b
console.log('o.p:', ins('o.p'));      // -> undefined
console.log('o.p.p:', ins('o.p.p'));  // -> undefined
console.log('p:', ins('p'));          // -> undefined
console.log(':', ins(''));            // -> undefined