当仅更新单个元素时,Meteor发送整个数组

时间:2015-05-15 01:56:29

标签: javascript arrays mongodb meteor livequery

我有一个具有10 x 10 2-d阵列字段的集合。当我更新此数组中的元素时,整个字段会在更改时发回给我。

I update one element of array and the entire array is sent back as changed

entire field is changed

有更好的方法来进行更新吗?

我是noSQL的新手,所以也许我只需要重新考虑我的数据库设置。这会有点令人沮丧,因为我喜欢能够将javascript对象直接映射到单个集合中的字段的简单性。但是,每次更新此内容时,我都不愿意丢失大约500字节的开销。

有没有办法强制Meteor通过更细粒度的更改来更新客户端?或者这是Meteor Livequery如何运作的限制?

谢谢!

1 个答案:

答案 0 :(得分:2)

问题是DDP仅支持顶级字段粒度的changed条消息。根据您当前的结构,对squares子字段的任何更改都会导致整个squares字段传输到客户端。

一种替代方法是将2D阵列的每个元素存储为单独的顶级字段。下面是一个完整的工作示例,使用-字符分隔元素:

var pool2squares = function(pool) {
  var squares = [];

  _.each(pool, function(value, key) {
    var parts = key.split('-');
    if (parts[0] === 'squares') {
      var i = Number(parts[1]);
      var j = Number(parts[2]);
      if (squares[i] == null)
        squares[i] = [];
      squares[i][j] = value;
    }
  });
  return squares;
};

Pools = new Mongo.Collection('pools', {
  transform: function(doc) {
    return _.extend(doc, {squares: pool2squares(doc)});
  }
});

Meteor.startup(function() {
  Pools.insert({
    'squares-0-0': 'a',
    'squares-0-1': 'b',
    'squares-1-0': 'c',
    'squares-1-1': 'd'
  });

  console.log(Pools.findOne().squares);
});

我们使用collection transform为每个squares文档添加Pool属性。上面的代码将打印出来:

[ [ 'a', 'b' ], [ 'c', 'd' ] ]

更新Pool

Pools.update(id, {$set: {'squares-3-4': 'hello'}});