Подтвердить что ты не робот

Преобразование строки в точечной нотации для получения ссылки на объект

Рассмотрим этот объект в javascript,

var obj = { a : { b: 1, c: 2 } };

учитывая строку "obj.a.b", как я могу получить объект, на который это ссылается, чтобы я мог изменить его значение? т.е. я хочу иметь возможность делать что-то вроде

obj.a.b = 5;
obj.a.c = 10;

где "obj.a.b" и "obj.a.c" являются строками (а не ссылками obj). Я столкнулся с этим сообщением, где я могу получить значение, которое строка точечной нотации ссылается на obj, но что мне нужно - это способ, которым я могу получить сам объект?

Вложение объекта может быть еще глубже, чем это. то есть, возможно,

var obj = { a: { b: 1, c : { d : 3, e : 4}, f: 5 } }
4b9b3361

Ответ 1

Чтобы получить значение, рассмотрим:

function ref(obj, str) {
    str = str.split(".");
    for (var i = 0; i < str.length; i++)
        obj = obj[str[i]];
    return obj;
}

var obj = { a: { b: 1, c : { d : 3, e : 4}, f: 5 } }
str = 'a.c.d'
ref(obj, str) // 3

или более причудливым способом, используя reduce:

function ref(obj, str) {
    return str.split(".").reduce(function(o, x) { return o[x] }, obj);
}

Возврат назначаемой ссылки на член объекта в javascript невозможен, вам нужно будет использовать следующую функцию:

function set(obj, str, val) {
    str = str.split(".");
    while (str.length > 1)
        obj = obj[str.shift()];
    return obj[str.shift()] = val;
}

var obj = { a: { b: 1, c : { d : 3, e : 4}, f: 5 } }
str = 'a.c.d'
set(obj, str, 99)
console.log(obj.a.c.d) // 99

или используйте ref, приведенные выше, чтобы получить ссылку на содержащий объект, а затем примените к нему оператор []:

parts = str.split(/\.(?=[^.]+$)/)  // Split "foo.bar.baz" into ["foo.bar", "baz"]
ref(obj, parts[0])[parts[1]] = 99

Ответ 2

Аналогично запросу thg435, но с аргументом проверяет и поддерживает уровни гнезд, где один из уровней предков еще не определен или не является объектом.

setObjByString = function(obj, str, val) {
    var keys, key;
    //make sure str is a string with length
    if (!str || !str.length || Object.prototype.toString.call(str) !== "[object String]") {
        return false;
    }
    if (obj !== Object(obj)) {
        //if it not an object, make it one
        obj = {};
    }
    keys = str.split(".");
    while (keys.length > 1) {
        key = keys.shift();
        if (obj !== Object(obj)) {
            //if it not an object, make it one
            obj = {};
        }
        if (!(key in obj)) {
            //if obj doesn't contain the key, add it and set it to an empty object
            obj[key] = {};
        }
        obj = obj[key];
    }
    return obj[keys[0]] = val;
};

Использование:

var obj;
setObjByString(obj, "a.b.c.d.e.f", "hello");

Ответ 3

Если этот javascript работает в браузере, вы можете получить доступ к этому объекту следующим образом:

window['obj']['a']['b'] = 5

Итак, для строки "obj.a.b" вам нужно разбить ее на .:

var s = "obj.a.b"
var e = s.split(".")
window[e[0]][e[1]][e[2]] = 5

Ответ 4

Ниже приведена простая оболочка класса dict:

class Dots(dict):
    def __init__(self, *args, **kargs):
            super(Dots, self).__init__(*args, **kargs)

    def __getitem__(self, key):
            try:
                    item = super(Dots, self).__getitem__(key)
            except KeyError:
                    item = Dots()
                    self.__setitem__(key, item)

            return Dots(item) if type(item) == dict else item

    def __setitem__(self, key, value):
            if type(value) == dict: value = Dots(value)
            super(Dots, self).__setitem__(key, value)

    __getattr__ = __getitem__
    __setattr__ = __setitem__

Пример:

>>> a = Dots()
>>> a.b.c = 123
>>> a.b.c
123
>>> a.b
{'c': 123}
>>> a
{'b': {'c': 123}}

Отсутствует ключ, создаваемый "на лету" как пустой Dots():

>>> if a.Missing: print "Exists"
...
>>> a
{'Missing': {}, 'b': {'c': 123}}

Ответ 5

var obj = { a : { b: 1, c: 2 } };
walkObject(obj,"a.b"); // 1

function walkObject( obj, path ){
  var parts = path.split("."), i=0, part;
  while (obj && (part=parts[i++])) obj=obj[part];
  return obj;
}

Или, если вам нравится ваш код:

function walkObject( o, path ){
  for (var a,p=path.split('.'),i=0; o&&(a=p[i++]); o=o[a]);
  return o;
}

Ответ 6

Возвращение назначаемой ссылки на объект-член невозможно в javascript. Вы можете назначить значение для элемента глубокого объекта точечной нотацией с помощью одной строки кода, как это.

new Function('_', 'val', '_.' + path + ' = val')(obj, value);

В вашем случае:

var obj = { a : { b: 1, c: 2 } };

new Function('_', 'val', '_.a.b' + ' = val')(obj, 5); // Now obj.a.b will be equal to 5