db.collection.findAndModify()
MongoDB com drivers
Esta página documenta um método mongosh
. Para ver o método equivalente em um driver MongoDB, consulte a página correspondente da sua linguagem de programação:
Definição
db.collection.findAndModify(document)
Importante
Método de mongosh obsoleto
Use
findOneAndUpdate()
,findOneAndDelete()
oufindOneAndReplace()
em vez disso.Atualiza e retorna um único documento. Por padrão, o documento devolvido não inclui as modificações feitas na atualização. Para retornar o documento com as modificações feitas na atualização, use a opção
new
.
Compatibilidade
Você pode utilizar o db.collection.findAndModify()
para implantações hospedadas nos seguintes ambientes:
MongoDB Atlas: o serviço totalmente gerenciado para implantações do MongoDB na nuvem
MongoDB Enterprise: a versão autogerenciada e baseada em assinatura do MongoDB
MongoDB Community: uma versão com código disponível, de uso gratuito e autogerenciada do MongoDB
Sintaxe
Alterado na versão 5.0.
O método findAndModify()
tem o seguinte formato:
db.collection.findAndModify({ query: <document>, sort: <document>, remove: <boolean>, update: <document or aggregation pipeline>, new: <boolean>, fields: <document>, upsert: <boolean>, bypassDocumentValidation: <boolean>, writeConcern: <document>, maxTimeMS: <integer>, collation: <document>, arrayFilters: [ <filterdocument1>, ... ], let: <document> // Added in MongoDB 5.0 });
O método db.collection.findAndModify()
usa um parâmetro de documento com os seguintes campos de documentos incorporados:
Parâmetro | Tipo | Descrição | ||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
query | documento | Opcional. Os critérios de seleção para a modificação. O campo Se não for especificado, o padrão será um documento vazio. Se o argumento de consulta não for um documento, a operação será executada. | ||||||||||||||||||
| documento | Opcional. Determina qual documento a operação atualiza se a query selecionar vários documentos. Se o argumento de classificação não for um documento, ocorrerá um erro com a operação. O MongoDB não armazena documentos em uma collection em uma ordem específica. Ao ordenar em um campo que contém valores duplicados, os documentos que contêm esses valores podem ser retornados em qualquer ordem. Se desejar uma ordem de classificação consistente, inclua pelo menos um campo em sua ordenação que contenha valores exclusivos. A maneira mais fácil de garantir isso é incluir o campo Consulte Consistência de classificação para obter mais informações. | ||||||||||||||||||
remove | booleano | Deve especificar o campo remove ou update . Remove o documento especificado no campo query . Define como true para remover o documento selecionado. O padrão é false . | ||||||||||||||||||
update | documento ou array | Deve especificar o campo
| ||||||||||||||||||
new | booleano | Opcional. Quando true , retorna o documento atualizado em vez do original. O padrão é false . | ||||||||||||||||||
fields | documento | Opcional. Um subconjunto de campos para retornar. O documento Se o argumento Para mais informações sobre projeção, consulte Projeção de | ||||||||||||||||||
upsert | booleano | Opcional. Usado em conjunto com o campo Quando
Para evitar várias upserts, certifique-se de que o(s) campo(s) de O padrão é | ||||||||||||||||||
bypassDocumentValidation | booleano | Opcional. Permite que Novo na versão 3.2. | ||||||||||||||||||
writeConcern | documento | Opcional. Um documento que expressa o write concern. Omitir para usar o write concern padrão. Não defina explicitamente a preocupação de gravação para a operação se for executada em uma transação. Para usar write concern com transações, consulte Transações e write concern. Novo na versão 3.2. | ||||||||||||||||||
maxTimeMS | non-negative integer | Opcional. Especifica um limite de tempo em milissegundos. Se você não especificar um valor para O MongoDB encerra as operações que excedem o limite de tempo alocado usando o mesmo mecanismo de | ||||||||||||||||||
collation | documento | Opcional. Especifica o agrupamento a ser usado para a operação. A colocação permite que os usuários especifiquem regras específicas do idioma para comparação de strings, como regras para letras maiúsculas e marcas de acento. A opção de agrupamento tem a seguinte sintaxe:
Ao especificar agrupamento, o campo Se o agrupamento não for especificado, mas a coleção tiver um agrupamento padrão (consulte Se nenhum agrupamento for especificado para a coleção ou para as operações, o MongoDB usa a comparação binária simples usada nas versões anteriores para comparações de strings. Você não pode especificar vários agrupamentos para uma operação. Por exemplo, você não pode especificar agrupamentos diferentes por campo ou, se estiver realizando uma busca com uma classificação, não poderá usar um agrupamento para a busca e outro para a classificação. Novidade na versão 3.4. | ||||||||||||||||||
arrayFilters | array | Opcional. Uma array de documentos de filtro que determina quais elementos da array modificar para uma operação de atualização em um campo da array. No documento de atualização, use o operador posicional filtrado O Você pode incluir o mesmo identificador várias vezes no documento de atualização; entretanto, para cada identificador distinto (
No entanto, você pode especificar condições compostas no mesmo identificador em um único documento de filtro, como nos exemplos a seguir:
Para obter exemplos, consulte Especificar Observação
Novidade na versão 3.6. | ||||||||||||||||||
documento | Opcional. Especifica um documento com uma lista de variáveis. Isso permite que você melhore a legibilidade do comando separando as variáveis do texto da query. A sintaxe do documento é:
A variável é definida para o valor retornado pela expressão e não pode ser alterada posteriormente. Para acessar o valor de uma variável no comando, use o prefixo de dois cifrões ( Para usar uma variável para filtrar os resultados, você deve acessar a variável dentro do operador Para um exemplo completo utilizando Novidades na versão 5.0. |
Retornar dados
Para remover operações, se a query corresponder a um documento, findAndModify()
retorna o documento removido. Se a query não corresponder a um documento para remover, findAndModify()
retornará null
.
Para operações de atualização, o findAndModify()
retorna um dos seguintes:
Se o parâmetro
new
não estiver configurado ou forfalse
:o documento de pré-modificação se a query corresponder a um documento;
caso contrário,
null
.
Se
new
fortrue
:o documento atualizado se a query retornar uma correspondência;
o documento inserido se
upsert: true
e nenhum documento corresponder à query;caso contrário,
null
.
Comportamento
fields
Projeção
Importante
Consistência de linguagem
Como parte de tornar a projeção find()
e findAndModify()
consistente com o estágio $project
da agregação,
A projeção
find()
efindAndModify()
pode aceitar expressões de agregação e sintaxe.O MongoDB impõe restrições adicionais em relação às projeções. Consulte Restrições de Projeção para detalhes.
A opção fields
usa um documento no seguinte formato:
{ field1: <value>, field2: <value> ... }
Projeção | Descrição |
---|---|
<field>: <1 or true> | Especifica a inclusão de um campo. Se você especificar um número inteiro diferente de zero para o valor de projeção, a operação tratará o valor como true . |
<field>: <0 or false> | Especifica a exclusão de um campo. |
"<field>.$": <1 or true> | Usa o operador de projeção de array Não disponível para visualizações. |
<field>: <array projection> | Usa os operadores de projeção de array ( Não disponível para visualizações. |
<field>: <aggregation expression> | Especifica o valor do campo projetado. Com o uso de expressões de agregação e sintaxe, incluindo o uso de literais e variáveis de agregação, você pode projetar novos campos ou projetar campos existentes com novos valores.
|
Especificação de campo incorporada
Para campos em documentos incorporados, você pode especificar o campo usando:
notação de pontos, por exemplo
"field.nestedfield": <value>
formato aninhado, por exemplo
{ field: { nestedfield: <value> } }
_id
Projeção de campo
O campo _id
é incluído nos documentos retornados por padrão, a menos que você especifique explicitamente _id: 0
na projeção para suprimir o campo.
Inclusão ou exclusão
Uma projection
não pode conter especificações de inclusão e exclusão, com exceção do campo _id
:
Em projeções que incluem explicitamente campos, o campo
_id
é o único campo que você pode excluir explicitamente.Em projeções que excluem explicitamente campos, o campo
_id
é o único campo que você pode incluir explicitamente; entretanto, o campo_id
é incluído por padrão.
Para obter mais informações sobre "projection", consulte também:
Upsert com índice exclusivo
Os upserts podem criar documentos duplicados, a menos que haja um índice único para evitar duplicatas.
Considere um exemplo em que nenhum documento com o nome Andy
existe e vários clientes emitem o seguinte comando ao mesmo tempo:
db.people.findAndModify( { query: { name: "Andy" }, update: { $inc: { score: 1 } }, upsert: true } )
Se todas as operações findOneAndUpdate()
terminarem a fase de query antes que qualquer cliente insira dados com êxito e não houver um índice exclusivo no campo name
, cada operação findOneAndUpdate()
poderá resultar em uma inserção, criando vários documentos com name: Andy
.
Um índice único no campo name
garante que apenas um documento seja criado. Com um índice único em vigor, as múltiplas operações findOneAndUpdate()
agora exibem o seguinte comportamento:
Exatamente uma operação
findOneAndUpdate()
inserirá com sucesso um novo documento.Outras operações
findOneAndUpdate()
atualizam o documento recém-inserido ou falham devido a uma colisão de chave exclusiva.Para que outras operações do
findOneAndUpdate()
atualizem o documento recém-inserido, todas as seguintes condições devem ser atendidas:A collection de destino tem um índice único que causaria um erro de chave duplicado.
A operação de atualização não é
updateMany
oumulti
éfalse
.A condição de correspondência de atualização é:
Um único predicado de igualdade. Por exemplo
{ "fieldA" : "valueA" }
Um E lógico de predicados de igualdade. Por exemplo
{ "fieldA" : "valueA", "fieldB" : "valueB" }
O campo no predicado de igualdade correspondem ao campo no padrão de chave de índice único.
A operação de atualização não modifica nenhum campo no padrão de chave de índice único.
A tabela a seguir mostra exemplos de operações upsert
que, quando ocorre uma colisão de chaves, resultam em uma atualização ou falha.
Padrão de chave de índice exclusivo | Operação de atualização | Resultado | ||||||
---|---|---|---|---|---|---|---|---|
|
| O campo score do documento correspondente é incrementado em 1. | ||||||
|
| A operação falha porque modifica o campo no padrão de chave de índice único ( name ). | ||||||
|
| A operação falha porque os campos de predicados de igualdade ( name , email ) não correspondem ao campo chave de índice (name ). |
Coleções fragmentadas
Para usar findAndModify
em uma collection fragmentada:
Se você só tiver como alvo um shard, pode usar uma chave de shard parcial no campo
query
ou,Você pode fornecer uma condição de igualdade em uma chave de shard completa no campo
query
.
Os documentos em uma coleção fragmentada podem não ter campos de chave de fragmento. Para direcionar um documento que não tem a chave de fragmento, você pode usar a correspondência de igualdade null
em conjunto com outra condição de filtro (como no campo _id
). Por exemplo:
{ _id: <value>, <shardkeyfield>: null } // _id of the document missing shard key
Modificação da chave de fragmento
Você pode atualizar o valor da chave de fragmento de um documento, a menos que o campo de chave de fragmento seja o campo de _id
imutável.
Aviso
Os documentos em coleções fragmentadas podem não ter os campos chave de fragmentado. Tome cuidado para evitar remover acidentalmente a chave de fragmento ao alterar o valor dela em um documento.
Para atualizar o valor da chave de fragmento existente com db.collection.findAndModify()
:
Você deve executar em um
mongos
. Não emita a operação diretamente no fragmento.Você deve executar em uma transação ou como uma gravação repetível.
Você deve incluir um filtro de igualdade na chave de shard completa.
Chave de fragmento ausente
Os documentos em coleções fragmentadas podem não ter os campos de chave de fragmento. Para usar db.collection.findAndModify()
para configurar a chave de fragmento ausente do documento:
Você deve executar em um
mongos
. Não emita a operação diretamente no fragmento.Você deve executar em uma transação ou como retryable write se o novo valor da chave de shard não for
null
.Você deve incluir um filtro de igualdade na chave de shard completa.
Dica
Como um valor de chave ausente é retornado como parte de uma correspondência de igualdade nula,
para evitar a atualização de uma chave de valor nulo, inclua
condições de consulta (como no campo _id
) conforme apropriado.
Veja também:
Validação do documento
O método db.collection.findAndModify()
adiciona compatibilidade para a opção bypassDocumentValidation
, que permite ignorar a validação de documentos ao inserir ou atualizar documentos em uma coleção com regras de validação.
Comparações com o update
método
Ao atualizar um documento, db.collection.findAndModify()
e o método updateOne()
operam de forma diferente:
Se vários documentos corresponderem aos critérios de atualização para
db.collection.findAndModify()
, você poderá especificar umasort
para fornecer alguma medida de controle sobre qual documento atualizar.updateOne()
atualiza o primeiro documento que corresponde.Por padrão,
db.collection.findAndModify()
retorna a versão pré-modificada do documento. Para obter o documento atualizado, use a opçãonew
.O método
updateOne()
retorna um objetoWriteResult()
que contém o status da operação.Para retornar o documento atualizado, use o método
find()
. No entanto, outras atualizações podem ter modificado o documento entre a atualização e a recuperação do documento. Além disso, se a atualização modificou apenas um único documento, mas vários documentos corresponderam, você precisará usar lógica adicional para identificar o documento atualizado.
Se modificar um único documento, ambos o db.collection.findAndModify()
e o método updateOne()
atualizam atomicamente o documento. Consulte Atomicidade e transações para obter mais detalhes sobre as interações e a ordem das operações desses métodos.
Transações
db.collection.findAndModify()
pode ser usado dentro de transações distribuídas.
Importante
Na maioria dos casos, uma transação distribuída incorre em um custo de desempenho maior do que as gravações de um único documento, e a disponibilidade de transações distribuídas não deve substituir o design eficaz do esquema. Em muitos cenários, o modelo de dados desnormalizado (documentos e arrays incorporados) continuará a ser ideal para seus dados e casos de uso. Ou seja, para muitos cenários, modelar seus dados adequadamente minimizará a necessidade de transações distribuídas.
Para considerações adicionais sobre o uso de transações (como limite de tempo de execução e limite de tamanho do oplog), consulte também Considerações de produção.
Inserção nas Transações
Você pode criar coleção e índices dentro de uma transação distribuída se a transação não for uma transação de gravação cross-fragmento.
db.collection.findAndModify()
com upsert: true
pode ser executado em uma coleção existente ou em uma coleção inexistente. Se for executada em uma coleção inexistente, a operação cria a coleção.
Write concerns e transações
Não defina explicitamente a preocupação de gravação para a operação se for executada em uma transação. Para usar write concern com transações, consulte Transações e write concern.
Entradas de oplog
Se uma operação db.collection.findAndModify()
conseguir localizar e modificar um documento, ela adicionará uma entrada no oplog (log de operações). Se a operação falhar ou não encontrar um documento para modificar, ela não adicionará uma entrada no oplog.
Exemplos
Atualização e devolução
O método a seguir atualiza e retorna um documento existente na collection de pessoas onde o documento corresponde aos critérios de query:
db.people.findAndModify({ query: { name: "Tom", state: "active", rating: { $gt: 10 } }, sort: { rating: 1 }, update: { $inc: { score: 1 } } })
Este método executa as seguintes ações:
A
query
localiza um documento na collectionpeople
onde o camponame
tem o valorTom
, o campostate
tem o valoractive
e o camporating
tem um valorgreater than
10.O
sort
classifica os resultados da query em ordem crescente. Se vários documentos atenderem à condiçãoquery
, o método selecionará para modificação o primeiro documento conforme ordenado por estesort
.A atualização
increments
do valor do camposcore
por 1.O método retorna o documento original (ou seja, pré-modificação) selecionado para esta atualização:
{ "_id" : ObjectId("50f1e2c99beb36a0f45c6453"), "name" : "Tom", "state" : "active", "rating" : 100, "score" : 5 } Para retornar o documento atualizado, adicione a opção
new:true
ao método.Se nenhum documento corresponder à condição
query
, o método retornaránull
.
Upsert
O método a seguir inclui a opção upsert: true
para a operação update
atualizar um documento correspondente ou, se não existir nenhum documento correspondente, criar um novo documento:
db.people.findAndModify({ query: { name: "Gus", state: "active", rating: 100 }, sort: { rating: 1 }, update: { $inc: { score: 1 } }, upsert: true })
Se encontrar um documento correspondente, o método executará uma atualização.
Se não encontrar um documento correspondente, o método criará um novo documento. Como o método incluiu a opção sort
, ele retorna um documento vazio { }
como o documento original (pré-modificação):
{ }
Se o método não incluiu uma opção sort
, o método retorna null
.
null
Retornar novo documento
O método seguinte inclui a opção upsert: true
e a opção new:true
. O método atualiza um documento correspondente e retorna o documento atualizado ou, se não existir nenhum documento correspondente, insere um documento e retorna o documento recém-inserido no campo value
.
No exemplo a seguir, nenhum documento na collection people
corresponde à condição query
:
db.people.findAndModify({ query: { name: "Pascal", state: "active", rating: 25 }, sort: { rating: 1 }, update: { $inc: { score: 1 } }, upsert: true, new: true })
O método retorna o documento recém-inserido:
{ "_id" : ObjectId("50f49ad6444c11ac2448a5d6"), "name" : "Pascal", "rating" : 25, "score" : 1, "state" : "active" }
Classificar e remover
Ao incluir uma especificação sort
no campo rating
, o exemplo seguinte remove da collection people
um único documento com o valor state
de active
e o menor rating
entre os documentos correspondentes:
db.people.findAndModify( { query: { state: "active" }, sort: { rating: 1 }, remove: true } )
O método retorna o documento excluído:
{ "_id" : ObjectId("52fba867ab5fdca1299674ad"), "name" : "XYZ123", "score" : 1, "state" : "active", "rating" : 3 }
Especifique o agrupamento
Novidade na versão 3.4.
A colocação permite que os usuários especifiquem regras específicas do idioma para comparação de strings, como regras para letras maiúsculas e marcas de acento.
Uma coleção myColl
possui os seguintes documentos:
{ _id: 1, category: "café", status: "A" } { _id: 2, category: "cafe", status: "a" } { _id: 3, category: "cafE", status: "a" }
A seguinte operação inclui a opção coleção:
db.myColl.findAndModify({ query: { category: "cafe", status: "a" }, sort: { category: 1 }, update: { $set: { status: "Updated" } }, collation: { locale: "fr", strength: 1 } });
A operação retorna o seguinte documento:
{ "_id" : 1, "category" : "café", "status" : "A" }
Especificar arrayFilters
para operações de atualização de array
Observação
arrayFilters
não está disponível para atualizações que usam um pipeline de agregação
Ao atualizar um campo de array, você pode especificar arrayFilters
que determinam quais elementos de array atualizar.
arrayFilters
Atualizar elementos que correspondem aos critérios
Observação
arrayFilters
não está disponível para atualizações que usam um pipeline de agregação
Crie uma collection students
com os seguintes documentos:
db.students.insertMany( [ { "_id" : 1, "grades" : [ 95, 92, 90 ] }, { "_id" : 2, "grades" : [ 98, 100, 102 ] }, { "_id" : 3, "grades" : [ 95, 110, 100 ] } ] )
Para atualizar todos os elementos que são maiores ou iguais a 100
no array grades
, utilize o operador posicional filtrado $[<identifier>]
com a opção arrayFilters
no método db.collection.findAndModify()
:
db.students.findAndModify({ query: { grades: { $gte: 100 } }, update: { $set: { "grades.$[element]" : 100 } }, arrayFilters: [ { "element": { $gte: 100 } } ] })
A operação atualiza o campo grades
para um único documento e, após a operação, a collection tem os seguintes documentos:
{ "_id" : 1, "grades" : [ 95, 92, 90 ] } { "_id" : 2, "grades" : [ 98, 100, 100 ] } { "_id" : 3, "grades" : [ 95, 110, 100 ] }
Atualizar elementos específicos de uma array de documentos
Observação
arrayFilters
não está disponível para atualizações que usam um pipeline de agregação
Crie uma collection students2
com os seguintes documentos:
db.students2.insertMany( [ { "_id" : 1, "grades" : [ { "grade" : 80, "mean" : 75, "std" : 6 }, { "grade" : 85, "mean" : 90, "std" : 4 }, { "grade" : 85, "mean" : 85, "std" : 6 } ] }, { "_id" : 2, "grades" : [ { "grade" : 90, "mean" : 75, "std" : 6 }, { "grade" : 87, "mean" : 90, "std" : 3 }, { "grade" : 85, "mean" : 85, "std" : 4 } ] } ] )
A operação a seguir localiza um documento em que o campo _id
é igual a 1
e usa o operador posicional filtrado $[<identifier>]
com arrayFilters
para atualizar o mean
para todos os elementos na array grades
onde a nota é maior maior ou igual a 85
.
db.students2.findAndModify({ query: { _id : 1 }, update: { $set: { "grades.$[elem].mean" : 100 } }, arrayFilters: [ { "elem.grade": { $gte: 85 } } ] })
A operação atualiza o campo grades
para um único documento e, após a operação, a collection tem os seguintes documentos:
{ "_id" : 1, "grades" : [ { "grade" : 80, "mean" : 75, "std" : 6 }, { "grade" : 85, "mean" : 100, "std" : 4 }, { "grade" : 85, "mean" : 100, "std" : 6 } ] } { "_id" : 2, "grades" : [ { "grade" : 90, "mean" : 75, "std" : 6 }, { "grade" : 87, "mean" : 90, "std" : 3 }, { "grade" : 85, "mean" : 85, "std" : 4 } ] }
Usar um aggregation pipeline para atualizações
db.collection.findAndModify()
pode aceitar um pipeline de agregação para a atualização. O pipeline pode consistir nas seguintes etapas:
$addFields
e seu alias$set
$replaceRoot
e seu nome alternativo$replaceWith
.
O uso do aggregation pipeline permite uma instrução de atualização mais expressiva, como atualizações condicionais Express com base em valores de campo atuais ou atualização de um campo usando o valor de outro(s) campo(s).
Por exemplo, criar uma collection students2
com os seguintes documentos:
db.students2.insertMany( [ { "_id" : 1, "grades" : [ { "grade" : 80, "mean" : 75, "std" : 6 }, { "grade" : 85, "mean" : 90, "std" : 4 }, { "grade" : 85, "mean" : 85, "std" : 6 } ] }, { "_id" : 2, "grades" : [ { "grade" : 90, "mean" : 75, "std" : 6 }, { "grade" : 87, "mean" : 90, "std" : 3 }, { "grade" : 85, "mean" : 85, "std" : 4 } ] } ] )
A operação a seguir localiza um documento em que o campo _id
é igual a 1
e usa um aggregation pipeline para calcular um novo total
de campo a partir do campo grades
:
db.students2.findAndModify( { query: { "_id" : 1 }, update: [ { $set: { "total" : { $sum: "$grades.grade" } } } ], // The $set stage is an alias for ``$addFields`` stage new: true } )
Observação
A operação retorna o documento atualizado:
{ "_id" : 1, "grades" : [ { "grade" : 80, "mean" : 75, "std" : 6 }, { "grade" : 85, "mean" : 90, "std" : 4 }, { "grade" : 85, "mean" : 85, "std" : 6 } ], "total" : 250 }
Usar variáveis em let
Novidades na versão 5.0.
Para definir variáveis que você pode acessar em outro lugar no comando, use a opção let .
Observação
Para filtrar resultados usando uma variável, você deve acessar a variável dentro do operador $expr
.
Criar uma coleção cakeFlavors
:
db.cakeFlavors.insertMany( [ { _id: 1, flavor: "chocolate" }, { _id: 2, flavor: "strawberry" }, { _id: 3, flavor: "cherry" } ] )
O exemplo a seguir define uma variável targetFlavor
em let
e usa a variável para alterar o sabor do bolo de cereja para laranja:
db.cakeFlavors.findAndModify( { query: { $expr: { $eq: [ "$flavor", "$$targetFlavor" ] } }, update: { flavor: "orange" }, let: { targetFlavor: "cherry" } } )