How to toggle a boolean field in one document with atomic operation?

前端 未结 2 1176
甜味超标
甜味超标 2020-12-06 16:13

Is there any way to toggle the boolean field of ONE document in MongoDB with atomic operation? Say, (In python)

cl.update({\"_id\": ...}, {\"$toggle\": {\"fi         


        
相关标签:
2条回答
  • 2020-12-06 17:00

    The SERVER-4362 issue is actually resolved now and you have the $bit update operator available. So along with it's xor argument you can now do this in an atomic action:

    cl.findOneAndUpdate( 
      { "_id": ...}, 
      { 
         "$bit": { 
             "field": { "xor": NumberInt(1) } 
         } 
      },
      { "returnNewDocument": true, "upsert": true }
    );
    

    So as long as the value of field is kept at 0 or 1 then a bitwise "flip" will result that makes the current value the opposite of what it was at the time of the modification.

    The .findOneAndUpdate() is not required, but just a way of demonstrating that the resulting value is different on every modification.

    0 讨论(0)
  • 2020-12-06 17:11

    Right now, I don't think it's possible to do this with one operation. The bitwise operators (http://www.mongodb.org/display/DOCS/Updating#Updating-%24bit) don't have a '$xor' yet although I've a patch for it.

    Right now the workaround I think think of is by always using '$inc':

    cl.update( { "_id": ...}, { '$inc' : { 'field' : 1 } } );

    Then instead of checking for true or false, you can do check whether an item is "true":

    cl.find( { "_id": ..., 'field' : { '$mod' : [ 2, 1 ] } );

    IE, you using the modulo operator to see whether it's even or uneven with even being "unset", and uneven being "set". If you want to have the oppposite behaviour (ie, find all items that don't have the flag set), then use

    [ 2, 0 ];

    0 讨论(0)
提交回复
热议问题