首頁 > 軟體

詳解Python中Addict模組的使用方法

2022-05-30 14:01:15

介紹

Addit 是一個Python模組,除了提供標準的字典語法外,Addit 生成的字典的值既可以使用屬性來獲取,也可以使用屬性進行設定。

這意味著你不用再寫這樣的字典了:

body = {
    'query': {
        'filtered': {
            'query': {
                'match': {'description': 'addictive'}
            },
            'filter': {
                'term': {'created_by': 'Mats'}
            }
        }
    }
}

相反,你只需編寫以下三行程式碼就能完成目的:

body = Dict()
body.query.filtered.query.match.description = 'addictive'
body.query.filtered.filter.term.created_by = 'Mats'

1.安裝

你可以通過 pip 安裝:

pip install addict

或通過 conda :

conda install addict -c conda-forge

Addit 在Python2.7+和Python3上都可以執行。

2.用法

Addict 繼承自字典,但在存取和設定其值方面更加靈活。使用 Addict 的字典是一種樂趣!

設定巢狀詞典的項是極其舒服的:

>>> from addict import Dict
>>> mapping = Dict()
>>> mapping.a.b.c.d.e = 2
>>> mapping
{'a': {'b': {'c': {'d': {'e': 2}}}}}

如果Dict是用任何可迭代值範例化的,它將遍歷並克隆這些值,然後寫入到對應的屬性及值中,比如:

>>> mapping = {'a': [{'b': 3}, {'b': 3}]}
>>> dictionary = Dict(mapping)
>>> dictionary.a[0].b
3

 mapping['a'] 不再與 dictionary['a'] 相同。

>>> mapping['a'] is dictionary['a']
False

當然,此特點僅限於建構函式,而不是在使用屬性或設定值時:

>>> a = Dict()
>>> b = [1, 2, 3]
>>> a.b = b
>>> a.b is b
True

3.要牢記的事情

記住, int 不是有效的屬性名,因此必須使用 get/setitem 語法 設定/獲取 非字串的 dict 鍵:

>>> addicted = Dict()
>>> addicted.a.b.c.d.e = 2
>>> addicted[2] = [1, 2, 3]
{2: [1, 2, 3], 'a': {'b': {'c': {'d': {'e': 2}}}}}

不過,你可以隨意混合使用這兩種語法:

>>> addicted.a.b['c'].d.e
2

4.屬性,如鍵、item等

Addit 不會讓你覆蓋 dict 的屬性,因此以下操作將不起作用:

>>> mapping = Dict()
>>> mapping.keys = 2
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
 File "addict/addict.py", line 53, in __setattr__
 raise AttributeError("'Dict' object attribute '%s' is read-only" % name)
AttributeError: 'Dict' object attribute 'keys' is read-only

不過,使用下面這種方式就可以:

>>> a = Dict()
>>> a['keys'] = 2
>>> a
{'keys': 2}
>>> a['keys']
2

5.預設值

對於不在字典中的鍵,Addit的行為如 defaultdict(Dict) ,因此丟失的鍵返回一個空的 Dict 而不是丟擲 KeyError 如果此行為不是所需的,則可以使用以下方式恢復丟擲KeyError:

>>> class DictNoDefault(Dict):
>>>   def __missing__(self, key):
>>>     raise KeyError(key)

但請注意,這樣會失去速記賦值功能(addicted.a.b.c.d.e = 2 )

6.轉化為普通字典

如果你覺得將 Addict 傳遞到其他函數或模組並不安全,請使用 to_dict() 方法,它返回會把 Addict 轉化為普通字典。

>>> regular_dict = my_addict.to_dict()
>>> regular_dict.a = 2
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
 AttributeError: 'dict' object has no attribute 'a'

當您希望在幾行程式碼中建立巢狀的字典,然後將其傳送到不同的函數或模組時,這非常適合:

body = Dict()
body.query.filtered.query.match.description = 'addictive'
body.query.filtered.filter.term.created_by = 'Mats'
third_party_module.search(query=body.to_dict())

7.計數

Dict輕鬆存取和修改深度巢狀屬性的能力使其成為計數的理想選擇。使用Addict,你還可以容易允許按多個級別計數,內部使用的原理是collections.Counter

比如以下資料:

data = [
    {'born': 1980, 'gender': 'M', 'eyes': 'green'},
    {'born': 1980, 'gender': 'F', 'eyes': 'green'},
    {'born': 1980, 'gender': 'M', 'eyes': 'blue'},
    {'born': 1980, 'gender': 'M', 'eyes': 'green'},
    {'born': 1980, 'gender': 'M', 'eyes': 'green'},
    {'born': 1980, 'gender': 'F', 'eyes': 'blue'},
    {'born': 1981, 'gender': 'M', 'eyes': 'blue'},
    {'born': 1981, 'gender': 'F', 'eyes': 'green'},
    {'born': 1981, 'gender': 'M', 'eyes': 'blue'},
    {'born': 1981, 'gender': 'F', 'eyes': 'blue'},
    {'born': 1981, 'gender': 'M', 'eyes': 'green'},
    {'born': 1981, 'gender': 'F', 'eyes': 'blue'}
]

如果你想計算有多少人出生在born性別的gender使用eyes眼睛,你可以很容易地計算出這些資訊:

counter = Dict()

for row in data:
    born = row['born']
    gender = row['gender']
    eyes = row['eyes']

    counter[born][gender][eyes] += 1 print(counter)

# 結果:{1980: {'M': {'blue': 1, 'green': 3}, 'F': {'blue': 1, 'green': 1}}, 1981: {'M': {'blue': 2, 'green': 1}, 'F': {'blue': 2, 'green': 1}}}

8.更新

普通字典的更新方式如下:

>>> d = {'a': {'b': 3}}
>>> d.update({'a': {'c': 4}})
>>> print(d)
{'a': {'c': 4}}

 addict 的更新方式如下,它會遞迴並實際更新巢狀的字典:

>>> D = Dict({'a': {'b': 3}})
>>> D.update({'a': {'c': 4}})
>>> print(D)
{'a': {'b': 3, 'c': 4}}

9.Addict 是怎麼來的

這個模組完全是從用Python建立Elasticsearch查詢的繁瑣過程中發展而來的。每當你發現自己在寫了很複雜的字典邏輯時,只要記住你沒有必要這樣做,使用 Addict 就行。

到此這篇關於詳解Python中Addict模組的使用方法的文章就介紹到這了,更多相關Python Addict模組內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com