Python 炫技操作(02):合并字典的七種方法
系列導(dǎo)讀:Python 炫技操作(01):條件語(yǔ)句的七種寫(xiě)法
Python 語(yǔ)言里有許多(而且是越來(lái)越多)的高級(jí)特性,是 Python 發(fā)燒友們非常喜歡的。在這些人的眼里,能夠?qū)懗瞿切┮话汩_(kāi)發(fā)者看不懂的高級(jí)特性,就是高手,就是大神。
但你要知道,在團(tuán)隊(duì)合作里,炫技是大忌。
為什么這么說(shuō)呢?我說(shuō)下自己的看法:
- 越簡(jiǎn)潔的代碼,越清晰的邏輯,就越不容易出錯(cuò);
- 在團(tuán)隊(duì)合作中,你的代碼不只有你在維護(hù),降低別人的閱讀/理解代碼邏輯的成本是一個(gè)良好的品德
- 簡(jiǎn)單的代碼,只會(huì)用到最基本的語(yǔ)法糖,復(fù)雜的高級(jí)特性,會(huì)有更多的依賴(lài)(如語(yǔ)言的版本)
該篇是「炫技系列」的第二篇內(nèi)容,在這個(gè)系列里,我將總結(jié)盤(pán)點(diǎn)一下,我所見(jiàn)過(guò)的那些炫技操作。在這里,如果你是 Python 發(fā)燒友,你可以學(xué)到一些寫(xiě)出超酷的代碼書(shū)寫(xiě)技巧。同時(shí),看了這些內(nèi)容,對(duì)你在閱讀別人的代碼時(shí),也許會(huì)有些幫助。
1. 最簡(jiǎn)單的原地更新
字典對(duì)象內(nèi)置了一個(gè) update 方法,用于把另一個(gè)字典更新到自己身上。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> profile.update(ext_info)
- >>> print(profile)
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
如果想使用 update 這種最簡(jiǎn)單、最地道原生的方法,但又不想更新到自己身上,而是生成一個(gè)新的對(duì)象,那請(qǐng)使用深拷貝。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> from copy import deepcopy
- >>>
- >>> full_profile = deepcopy(profile)
- >>> full_profile.update(ext_info)
- >>>
- >>> print(full_profile)
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
- >>> print(profile)
- {"name": "xiaoming", "age": 27}
2. 先解包再合并字典
使用 ** 可以解包字典,解包完后再使用 dict 或者 {} 就可以合并。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> full_profile01 = {**profile, **ext_info}
- >>> print(full_profile01)
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
- >>>
- >>> full_profile02 = dict(**profile, **ext_info)
- >>> print(full_profile02)
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
若你不知道 dict(**profile, **ext_info) 做了啥,你可以將它等價(jià)于
- >>> dict((("name", "xiaoming"), ("age", 27), ("gender", "male")))
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
3. 借助 itertools
在 Python 里有一個(gè)非常強(qiáng)大的內(nèi)置模塊,它專(zhuān)門(mén)用于操作可迭代對(duì)象。
正好我們字典也是可迭代對(duì)象,自然就可以想到,可以使用 itertools.chain() 函數(shù)先將多個(gè)字典(可迭代對(duì)象)串聯(lián)起來(lái),組成一個(gè)更大的可迭代對(duì)象,然后再使用 dict 轉(zhuǎn)成字典。
- >>> import itertools
- >>>
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>>
- >>> dict(itertools.chain(profile.items(), ext_info.items()))
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
4. 借助 ChainMap
如果可以引入一個(gè)輔助包,那我就再提一個(gè), ChainMap 也可以達(dá)到和 itertools 同樣的效果。
- >>> from collections import ChainMap
- >>>
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> dict(ChainMap(profile, ext_info))
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
使用 ChainMap 有一點(diǎn)需要注意,當(dāng)字典間有重復(fù)的鍵時(shí),只會(huì)取第一個(gè)值,排在后面的鍵值并不會(huì)更新掉前面的(使用 itertools 就不會(huì)有這個(gè)問(wèn)題)。
- >>> from collections import ChainMap
- >>>
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info={"age": 30}
- >>> dict(ChainMap(profile, ext_info))
- {'name': 'xiaoming', 'age': 27}
5. 使用dict.items() 合并
在 Python 3.9 之前,其實(shí)就已經(jīng)有 | 操作符了,只不過(guò)它通常用于對(duì)集合(set)取并集。
利用這一點(diǎn),也可以將它用于字典的合并,只不過(guò)得繞個(gè)彎子,有點(diǎn)不好理解。
你得先利用 items 方法將 dict 轉(zhuǎn)成 dict_items,再對(duì)這兩個(gè) dict_items 取并集,最后利用 dict 函數(shù),轉(zhuǎn)成字典。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> full_profile = dict(profile.items() | ext_info.items())
- >>> full_profile
- {'gender': 'male', 'age': 27, 'name': 'xiaoming'}
當(dāng)然了,你如果嫌這樣太麻煩,也可以簡(jiǎn)單點(diǎn),直接使用 list 函數(shù)再合并(示例為 Python 3.x )
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> dict(list(profile.items()) + list(ext_info.items()))
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
若你在 Python 2.x 下,可以直接省去 list 函數(shù)。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> dict(profile.items() + ext_info.items())
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
6. 最酷炫的字典解析式
Python 里對(duì)于生成列表、集合、字典,有一套非常 Pythonnic 的寫(xiě)法。
那就是列表解析式,集合解析式和字典解析式,通常是 Python 發(fā)燒友的最?lèi)?ài),那么今天的主題:字典合并,字典解析式還能否勝任呢?
當(dāng)然可以,具體示例代碼如下:
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> {k:v for d in [profile, ext_info] for k,v in d.items()}
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
7. Python 3.9 新特性
在 2 月份發(fā)布的 Python 3.9.04a 版本中,新增了一個(gè)抓眼球的新操作符操作符:|, PEP584 將它稱(chēng)之為合并操作符(Union Operator),用它可以很直觀地合并多個(gè)字典。
- >>> profile = {"name": "xiaoming", "age": 27}
- >>> ext_info = {"gender": "male"}
- >>>
- >>> profile | ext_info
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
- >>>
- >>> ext_info | profile
- {'gender': 'male', 'name': 'xiaoming', 'age': 27}
- >>>
- >>>
除了 | 操作符之外,還有另外一個(gè)操作符 |=,類(lèi)似于原地更新。
- >>> ext_info |= profile
- >>> ext_info
- {'gender': 'male', 'name': 'xiaoming', 'age': 27}
- >>>
- >>>
- >>> profile |= ext_info
- >>> profile
- {'name': 'xiaoming', 'age': 27, 'gender': 'male'}
看到這里,有沒(méi)有漲姿勢(shì)了,學(xué)了這么久的 Python ,沒(méi)想到合并字典還有這么多的方法。本篇文章的主旨,并不在于讓你全部掌握這 7 種合并字典的方法,實(shí)際上,你只要選用一種最順手的方式即可。
但是在協(xié)同工作中,或者在閱讀他人代碼時(shí),你不可避免地會(huì)碰到各式各樣的寫(xiě)法,這時(shí)候你能下意識(shí)的知道這是在做合并字典的操作,那這篇文章就是有意義的。