Расширенное агрегирование данных: подсчет среднего в коллекциях MongoDB - PullRequest
3 голосов
/ 06 июня 2011

У меня есть коллекция документов, таких как:

{
"browser" : "firefox",
"version" : "4.0.1"
}

{
"browser" : "firefox",
"version" : "3.6.2"
}

{
"browser" : "ie",
"version" : "8.0"
}

Как посчитать среднее значение для всех браузеров, чтобы результаты были:

global firefox: 66%
global ie: 33%

precise firefox:
4.0.1: 50%
3.6.3: 50%

Сложность в том, что я не хочу предоставлять все версии Firefox, доступные в массиве. Запрос MongoDB должен найти все отдельные версии в коллекции и подсчитать среднее для всех из них.

Заранее спасибо!

1 Ответ

6 голосов
/ 07 июня 2011

Вот решение, которое производит вашу статистику с чистыми числами (например, 0,5 вместо 50%):

var m = function() {
  emit('global', this.browser);
  emit('local', [this.browser, this.version]);
};

var r = function(key, values) {
  var global={}, local={}, total=0, i, j, x;
  if (key == 'global') {
    values.forEach(function(v) {
      global[v] = (global[v]||0) + 1;
      total += 1;
    });
    for (i in global) { global[i] = global[i] / total; }
    return global;
  } else if (key == 'local') {
    values.forEach(function(v) {
      if (!local[v[0]]) { local[v[0]] = {}; }
      x = local[v[0]];
      x[v[1]] = (x[v[1]]||0) + 1;
    });
    for (i in local) {
      total = 0;
      x = local[i];
      for (j in x) { total += x[j]; }
      for (j in x) { x[j] = x[j] / total; }
    }
    return local;
  };
};

db.browsers.mapReduce(m, r, {out:'bout'});
db.bout.find();
// => { "_id" : "global", "value" : { "firefox" : 0.6666666666666666, "ie" : 0.3333333333333333 } }
// => { "_id" : "local", "value" : { "firefox" : { "4.0.1" : 0.5, "3.6.2" : 0.5 }, "ie" : { "8.0" : 1 } } }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...