有没有办法将元数据添加到JavaScript对象?

时间:2012-07-31 13:13:18

标签: javascript node.js v8

我想将元数据的键值对添加到任意JavaScript对象。此元数据不应影响不了解元数据的代码,例如

JSON.stringify(obj) === JSON.stringify(obj.WithMetaData('key', 'value'))

MetaData感知代码应该能够通过密钥检索数据,即

obj.WithMetaData('key', 'value').GetMetaData('key') === 'value'

有什么办法吗 - 在node.js中?如果是这样,它是否适用于内置类型,例如String 甚至Number ? (编辑考虑一下,我并不关心真正的原语,就像数字一样,但对于 string 实例来说这样会很好。)< / p>

一些背景:我要做的是缓存从对象本身派生的对象,以便

  • 到元数据无意识代码,元数据丰富对象看起来与原始对象w / o meta相同
  • 需要派生值的代码可以从已经缓存的元数据中获取
  • 缓存将与对象一起收集垃圾

另一种方法是在某处存储带有缓存的哈希表,但是你永远不知道对象何时被垃圾收集。每个对象实例都必须手动处理,以便缓存不会泄漏。

(btw clojure有这个功能:http://clojure.org/metadata

4 个答案:

答案 0 :(得分:7)

您可以使用ECMA5的新对象属性API来存储不会在枚举中显示但仍可检索的对象的属性。

var myObj = {};
myObj.real_property = 'hello';
Object.defineProperty(myObj, 'meta_property', {value: 'some meta value'});
for (var i in myObj)
    alert(i+' = '+myObj[i]); //only one property - @real_property
alert(myObj.meta_property); //"some meta value"

此处提供更多信息:link

但是,您无法在原始类型(如字符串或数字)上执行此操作,只能在复杂类型上执行此操作。

[编辑]

另一种方法可能是利用数据类型的原型来存储元数据。 (警告,黑客前进)。所以对于字符串:

String.prototype.meta = {};
String.prototype.addMeta = function(name, val) { this.meta[name] = val; }
String.prototype.getMeta = function(name) { return this.meta[name]; };
var str = 'some string value';
str.addMeta('meta', 'val');
alert(str.getMeta('meta'));

然而,这显然并不理想。首先,如果字符串被收集或别名(因为简单数据类型是按值复制而不是引用),您将丢失此元数据。说实话,只有第一种方法在现实环境中有任何里程。

答案 1 :(得分:4)

ES6规范引入了Map和WeakMap。您可以通过运行node --harmony并在Chrome中启用实验性JavaScript标记来启用节点中的这些标记(默认情况下也在Firefox中)。 Maps和WeakMaps允许将对象用作键,可用于存储关于对象的元数据,这些对象在没有访问特定map / weakmap的情况下对任何人都不可见。这是我现在经常使用的模式:

function createStorage(creator){
  creator = creator || Object.create.bind(null, null, {});
  var map = new Map;
  return function storage(o, v){
    if (1 in arguments) {
      map.set(o, v);
    } else {
      v = map.get(o);
      if (v == null) {
        v = creator(o);
        map.set(o, v);
      }
    }
    return v;
  };
}

使用简单而有力:

var _ = createStorage();

_(someObject).meta= 'secret';
_(5).meta = [5];
var five = new Number(5);
_(five).meta = 'five';

console.log(_(someObject).name);
console.log(_(5).meta);
console.log(_(five).meta);

它还有助于将实现与接口分离的一些有趣用途:

var _ = createStorage(function(o){ return new Backing(o) });

function Backing(o){
  this.facade = o;
}
Backing.prototype.doesStuff = function(){
  return 'real value';
}

function Facade(){
  _(this);
}
Facade.prototype.doSomething = function doSomething(){
  return _(this).doesStuff();
}

答案 2 :(得分:1)

JSON中没有“评论”系统。您可以期望的最好的方法是添加名称不太可能的属性,并添加包含元数据的密钥。然后,如果您知道元数据,则可以将元数据读回来,但其他设置只会将其视为另一个属性。如果有人使用for..in ...

答案 3 :(得分:1)

您可以将元数据添加为“私有”变量!?

var Obj = function (meta) {
    var meta = meta;
    this.getMetaData = function (key) {
        //do something with the meta object
        return meta;
    };
};
var ins_ob = new Obj({meta:'meta'});
var ins_ob2 = new Obj();
if(JSON.stringify(ins_ob) === JSON.stringify(ins_ob2)) {
    console.log('hoorai');
};