Обновите коллекцию MongoDB, используя $ toLower - PullRequest
31 голосов
/ 24 февраля 2012

У меня есть коллекция MongoDB, содержащая имена пользователей.Имена пользователей содержат как строчные, так и прописные буквы.

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

Я пробовал этот скрипт, но он не работал

db.myCollection.find().forEach(
 function(e) {
 e.UserName = $toLower(e.UserName);
 db.myCollection.save(e);
 }
)

Ответы [ 6 ]

58 голосов
/ 24 февраля 2012

MongoDB не имеет понятия $toLower как команды. Решение состоит в том, чтобы запустить большой цикл for для данных и выпускать обновления по отдельности.

Вы можете сделать это в любом драйвере или из оболочки:

db.myCollection.find().forEach(
  function(e) {
    e.UserName = e.UserName.toLowerCase();
    db.myCollection.save(e);
  }
)

Вы также можете заменить сохранение атомным обновлением:

db.myCollection.update({_id: e._id}, {$set: {UserName: e.UserName.toLowerCase() } })

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


РЕДАКТИРОВАТЬ: Ремон поднимает хорошую мысль. Команда $toLower существует как часть структуры агрегации, но это не имеет ничего общего с обновлением. Документация для обновления здесь .

2 голосов
/ 31 мая 2016

Очень похожее решение, но это сработало у меня в новой монго 3,2 Выполните следующее в Mongo Shell или эквивалентных инструментах БД, таких как MongoChef!

db.tag.find({hashtag :{ $exists:true}}).forEach(
 function(e) {
   e.hashtag = e.hashtag.toLowerCase();
   db.tag.save(e);
});
1 голос
/ 23 июня 2019

Начиная с Mongo 4.2, db.collection.update() можно принять конвейер агрегации, наконец, разрешив обновление поля на основе его собственного значения:

// { username: "Hello World" }
db.collection.update(
  {},
  [{ $set: { username: { $toLower: "$username" } } }],
  { multi: true }
)
// { username: "hello world" }
  • Первая часть {} - это запрос на совпадение, отфильтровывающий, какие документы обновлять (в данном случае все документы).

  • Вторая часть [{ $set: { username: { $toLower: "$username" } } }], - это конвейер агрегации обновлений (обратите внимание на квадратные скобки, обозначающие использование конвейера агрегации):

    • $set - это новый оператор агрегирования, который в этом случае изменяет значение для "username".
    • Используя $toLower, мы изменяем значение "username" его строчной версией.
  • Не забудьте { multi: true }, иначе будет обновлен только первый соответствующий документ.

0 голосов
/ 05 апреля 2019

Немного опоздал на вечеринку, но приведенный ниже ответ очень хорошо работает с mongo 3.4 и выше. Сначала получите только те записи, которые имеют другой регистр, и обновите только эти записи оптом.Производительность этого запроса в несколько раз лучше

var bulk = db.myCollection.initializeUnorderedBulkOp();
var count = 0
db.myCollection.find({userId:{$regex:'.*[A-Z]'}}).forEach(function(e) {
 var newId = e.userId.toLowerCase();   
    bulk.find({_id:e._id}).updateOne({$set:{userId: newId}})
    count++
    if (count % 500 === 0) {
        bulk.execute();
        bulk = db.myCollection.initializeUnorderedBulkOp();
        count = 0;
    }
})
if (count > 0)  bulk.execute();
0 голосов
/ 20 мая 2015

С принятым решением я знаю, что очень просто сделать то же самое для массива элементов, на всякий случай

db.myCollection.find().forEach(
   function(e) {
      for(var i = 0; i < e.articles.length; i++) { 
          e.articles[i] = e.articles[i].toLowerCase(); 
      }
      db.myCollection.save(e); 
   }
)
0 голосов
/ 17 марта 2015

Просто примечание, чтобы убедиться, что поле существует для всех записей в вашей коллекции. Если нет, вам понадобится оператор if, например:

if (e.UserName) e.UserName = e.UserName.toLowerCase();
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...