Есть ли способ добавить в список один запрос? - PullRequest
6 голосов
/ 12 августа 2010

Я знаю, что этот вопрос уже задавался, но это другой сценарий. Я хотел бы иметь такую ​​коллекцию:

{
    "_id" : ObjectId("4c28f62cbf8544c60506f11d"),
    "pk": 1,
    "forums": [{
        "pk": 1,
        "thread_count": 10, 
        "post_count": 20,
    }, {
        "pk": 2,
        "thread_count": 5, 
        "post_count": 24,
    }]
}

Что я хочу сделать, это добавить элемент «форум», увеличить счетчики или добавить элемент, если он не существует.

Например, сделать что-то вроде этого (надеюсь, это имеет смысл):

db.mycollection.update({
    "pk": 3,
    "forums.pk": 2
}, {
    "$inc": {"forums.$.thread_count": 1},
    "$inc": {"forums.$.post_count": 1},
}, true)

и имеют:

{
    "_id" : ObjectId("4c28f62cbf8544c60506f11d"),
    "pk": 1,
    "forums": [{
        "pk": 1,
        "thread_count": 10, 
        "post_count": 20,
    }, {
        "pk": 2,
        "thread_count": 5, 
        "post_count": 24,
    }]
},
{
    "_id" : ObjectId("4c28f62cbf8544c60506f11e"),
    "pk": 3,
    "forums": [{
        "pk": 2,
        "thread_count": 1, 
        "post_count": 1,
    }]
}

Конечно, я могу сделать это в три этапа:

  1. Добавить всю коллекцию с новым предметом
  2. addToSet элемент форума в списке
  3. увеличение счетчиков элементов форума с позиционным оператором

То есть:

db.mycollection.update({pk:3}, {pk:3}, true)
db.mycollection.update({pk:3}, {$addToSet: {forums: {pk:2}}})
db.mycollection.update({pk:3, 'forums.pk': 2}, {$inc: {'forums.$.thread_counter': 1, {'forums.$.post_counter': 1}})

Вам известен более эффективный способ сделать это? ТИА, Джермано

1 Ответ

10 голосов
/ 03 ноября 2010

Как вы, возможно, обнаружили, позиционный оператор нельзя использовать в upserts:

Позиционный оператор нельзя комбинировать с upsert, поскольку для него требуется соответствующий элемент массива. Если ваше обновление приводит к вставке, то «$» будет буквально использоваться в качестве имени поля.

Таким образом, вы не сможете достичь желаемого результата за один запрос.

У вас есть , чтобы отделить создание документа от обновления счетчика. Ваше собственное решение находится на правильном пути. Это может быть сжато в следующие два запроса:

// optionally create the document, including the array
db.mycollection.update({pk:3}, {$addToSet: {forums: {pk:2}}}, true)

// update the counters in the array item
db.mycollection.update({pk:3, 'forums.pk': 2}, {$inc: {'forums.$.thread_counter': 1, 'forums.$.post_counter': 1}})
...