сжатие иерархий объектов в JavaScript - PullRequest
10 голосов
/ 08 июня 2009

Существует ли общий подход к "сжатию" вложенных объектов до одного уровня:

var myObj = {
    a: "hello",
    b: {
        c: "world"
    }
}

compress(myObj) == {
    a: "hello",
    b_c: "world"
}

Я полагаю, что будет некоторая рекурсия, но я решил, что мне не нужно изобретать велосипед здесь ...!?

Ответы [ 3 ]

21 голосов
/ 08 июня 2009
function flatten(obj, includePrototype, into, prefix) {
    into = into || {};
    prefix = prefix || "";

    for (var k in obj) {
        if (includePrototype || obj.hasOwnProperty(k)) {
            var prop = obj[k];
            if (prop && typeof prop === "object" &&
                !(prop instanceof Date || prop instanceof RegExp)) {
                flatten(prop, includePrototype, into, prefix + k + "_");
            }
            else {
                into[prefix + k] = prop;
            }
        }
    }

    return into;
}

Вы можете включить унаследованные элементы, передав true во второй параметр.

Несколько предостережений:

  • рекурсивные объекты не будут работать. Например:

    var o = { a: "foo" };
    o.b = o;
    flatten(o);
    

    будет повторяться до тех пор, пока не сгенерирует исключение.

  • Как и ответ ruquay, он вытягивает элементы массива так же, как обычные свойства объекта. Если вы хотите сохранить массивы нетронутыми, добавьте "|| prop instanceof Array" к исключениям.

  • Если вы вызовете это для объектов из другого окна или фрейма, даты и регулярные выражения не будут включены, поскольку instanceof не будет работать должным образом. Вы можете исправить это, заменив метод по умолчанию toString следующим образом:

    Object.prototype.toString.call(prop) === "[object Date]"
    Object.prototype.toString.call(prop) === "[object RegExp]"
    Object.prototype.toString.call(prop) === "[object Array]"
    
4 голосов
/ 08 июня 2009

Вот быстрый, но будьте осторожны, б / с он будет не работать с массивами и нулевыми значениями (б / с их typeof возвращает «объект»).

var flatten = function(obj, prefix) {
  if(typeof prefix === "undefined") {
    prefix = "";
  }
  var copy = {};
  for (var p in obj) {
    if(obj.hasOwnProperty(p)) {
      if(typeof obj[p] === "object") {
        var tmp = flatten(obj[p], p + "_");
        for(var q in tmp) {
          if(tmp.hasOwnProperty(q)) {
            copy[prefix + q] = tmp[q];
          }
        }
      }
      else {
        copy[prefix + p] = obj[p];
      }
    }
  }
  return copy;
}

var myObj = {
  a: "level 1",
  b: {
    a: "level 2",
    b: {
      a: "level 3",
      b: "level 3"
    }
  }
}

var flattened = flatten(myObj);
2 голосов
/ 04 августа 2011

Вот быстрая версия CoffeeScript , основанная на Ответ Мэтью Крамли (я не использовал includePrototype, так как мне это не нужно):

flatten = (obj, into = {}, prefix = '', sep = '_') ->
  for own key, prop of obj
    if typeof prop is 'object' and prop not instanceof Date and prop not instanceof RegExp
      flatten prop, into, prefix + key + sep, sep
    else
      into[prefix + key] = prop
  into

И базовая версия без платы, которая, несомненно, выйдет из строя при многократных разделителях и других подобных хитростях:

unflatten = (obj, into = {}, sep = '_') ->
  for own key, prop of obj
    subKeys = key.split sep
    sub = into
    sub = (sub[subKey] or= {}) for subKey in subKeys[...-1]
    sub[subKeys.pop()] = prop
  into

FWIW, я использую эти функции для вставки графов объектов в Хэши Redis , которые поддерживают только одну глубину пар ключ / значение.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...