How to get ordered dictionaries in pymongo?

前端 未结 4 751
忘了有多久
忘了有多久 2020-11-30 09:59

I am trying get ordered dictionaries in Pymongo. I have read it can be done with bson.son.Son. The Docs are Here

However, I can\'t seem to make it work. There is

相关标签:
4条回答
  • 2020-11-30 10:31

    You can use bson.son.SON or OrderedDict to store ordered dict.

    And retrive data with as_class=OrderedDict option.

    Here is an example:

    from collections import OrderedDict
    from pymongo import MongoClient
    import bson
    
    client = MongoClient()
    sample_db = client['sample']
    test_col = sample_db['test']
    
    test_col.drop()
    
    data = OrderedDict([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert(data)
    print(list(test_col.find({}, {'_id': 0}, as_class=OrderedDict)))
    
    test_col.drop()
    
    data = bson.son.SON([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert(data)
    print(list(test_col.find({}, {'_id': 0}, as_class=OrderedDict)))
    

    Output:

    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    
    0 讨论(0)
  • 2020-11-30 10:32

    In PyMongo v3.2 insert() has been deprecated and in this example it should be replaced with insert_one(). Updated code is below:

    from collections import OrderedDict
    from pymongo import MongoClient
    import bson
    
    client = MongoClient(document_class=OrderedDict)
    sample_db = client['sample']
    test_col = sample_db['test']
    
    test_col.drop()
    
    data = OrderedDict([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert_one(data)
    print(list(test_col.find({}, {'_id': 0})))
    
    test_col.drop()
    
    data = bson.son.SON([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert_one(data)
    print(list(test_col.find({}, {'_id': 0})))
    

    Output:

    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    
    0 讨论(0)
  • 2020-11-30 10:33

    This solution above is correct for older versions of MongoDB and the pymongo driver but it no longer works with pymongo3 and MongoDB3+ You now need to add document_class=OrderedDict to the MongoClient constructor. Modifying the above answer for pymongo3 compatibility.

    from collections import OrderedDict
    from pymongo import MongoClient
    import bson
    
    client = MongoClient(document_class=OrderedDict)
    sample_db = client['sample']
    test_col = sample_db['test']
    
    test_col.drop()
    
    data = OrderedDict([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert(data)
    print(list(test_col.find({}, {'_id': 0})))
    
    test_col.drop()
    
    data = bson.son.SON([("one", 1), ("two", 2), ("three", 3), ("four", 4)])
    test_col.insert(data)
    print(list(test_col.find({}, {'_id': 0})))
    

    Output:

    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    [OrderedDict([(u'one', 1), (u'two', 2), (u'three', 3), (u'four', 4)])]
    
    0 讨论(0)
  • 2020-11-30 10:45

    A standard find() in PyMongo will not return an object who's fields are in the same order as that object, if you retrieved it via mongo shell.

    This is because, the default type returned is a Dict and the order is not defined.

    You can use SON as suggested. Here is how I did it. Now the field order will be respected.

    This is for pymongo==3.4.0

    from bson.codec_options import CodecOptions
    from bson.son import SON
    
    opts = CodecOptions(document_class=SON)
    collection_son = mongo.db.collection.with_options(codec_options=opts)
    
    collection_son.find_one({"imsid": '12345'})
    
    0 讨论(0)
提交回复
热议问题