解讀Python中字典的key都可以是什么
Python字典的key都可以是什么
答
一個(gè)對(duì)象能不能作為字典的key,就取決于其有沒有__hash__方法。所以所有python自帶類型中,除了list、dict、set和內(nèi)部至少帶有上述三種類型之一的tuple之外,其余的對(duì)象都能當(dāng)key。
比如數(shù)值/字符串/完全不可變的元祖/函數(shù)(內(nèi)建或自定義)/類(內(nèi)建或自定義)/方法/包等等你能拿出手的,不過有的實(shí)際意義不高。還有數(shù)值型要注意,因?yàn)閮蓚€(gè)不同的相等數(shù)字可以有相同的哈希值,比如1和1.0。
解釋
代碼版本:3.6.3;文檔版本:3.6.6
Unlike sequences, which are indexed by a range of numbers, dictionaries are indexed by keys, which can be any immutable type; strings and numbers can always be keys. Tuples can be used as keys if they contain only strings, numbers, or tuples; if a tuple contains any mutable object either directly or indirectly, it cannot be used as a key. You can’t use lists as keys, since lists can be modified in place using index assignments, slice assignments, or methods like append()and extend().
字典的鍵可以是任意不可變類型,需要注意的是tuple元組作為鍵時(shí),其中不能以任何方式包含可變對(duì)象。
那。。到底什么樣的是不可變類型呢?不可能給對(duì)象專門標(biāo)注一個(gè)屬性是可變類型還是不可變類型啊,這沒有任何其他意義,一定是通過其他途徑實(shí)現(xiàn)的。把list當(dāng)做鍵試一下
a = [1, 2, 3] d = {a: a} ? ? # 第二行報(bào)錯(cuò): # TypeError: unhashable type: 'list'
報(bào)錯(cuò)說list類型是不可哈希的,噢,原來是靠能不能hash來判斷的,另外文檔下面接著說同一字典中每個(gè)鍵都是唯一的,正好每個(gè)對(duì)象的哈希值也是唯一的,對(duì)應(yīng)的很好。
It is best to think of a dictionary as an unordered set of key: value pairs, with the requirement that the keys are unique (within one dictionary).
查看源代碼可以看到object對(duì)象是定義了__hash__方法的,
而list、set和dict都把__hash__賦值為None了
# 部分源碼 ? class object: ? ? """ The most base type """ ? ? ? def __hash__(self, *args, **kwargs): ?# real signature unknown ? ? ? ? """ Return hash(self). """ ? ? ? ? pass ? ? class list(object): ? ? __hash__ = None ? ? class set(object): ? ? __hash__ = None ? ? class dict(object): ? ? __hash__ = None
那這樣的話。。。我給他加一個(gè)hash不就能當(dāng)字典的key了,key不就是可變的了。
注意
此處只是我跟著想法隨便試,真的應(yīng)用場(chǎng)景不要用可變類型作為字典的key。
class MyList(list): ? ? """比普通的list多一個(gè)__hash__方法""" ? ? ? def __hash__(self): ? ? ? ? # 不能返回hash(self) ? ? ? ? # hash(self)會(huì)調(diào)用self的本方法,再調(diào)用回去,那就沒完了(RecursionError) ? ? ? ? # 用的時(shí)候要注意實(shí)例中至少有一個(gè)元素,不然0怎么取(IndexError) ? ? ? ? return hash(self[0]) ? ? l1 = MyList([1, 2]) ?# print(l1) -> [1, 2] d = {l1: 'Can?'} print(d) ?# --> ?{[1, 2]: 'Can?'} l1.append(3) print(d) ?# {[1, 2, 3]: 'Can?'} print(d[l1]) ?# --> ?Can?
到這里就可以肯定的說,一個(gè)對(duì)象能不能作為字典的key,就取決于其有沒有__hash__方法。所以所有python自帶類型中,目前我已知的除了list、dict、set和內(nèi)部帶有以上三種類型的tuple之外,其余的對(duì)象都能當(dāng)key。而我們自己定義的類,一般情況下都直接間接的和object有關(guān),都帶有__hash__方法。
另外我想到,既然字典的鍵是唯一的,而哈希值也是唯一的,這么巧,鍵的唯一性不會(huì)就是用哈希值來確定的吧?我上一個(gè)例子中__hash__方法返回的是0號(hào)元素的哈希值,那我直接用相同哈希值的對(duì)象是不是就能改變那本來不屬于它的字典值呢?
class MyList(list): ? ? def __hash__(self): ? ? ? ? return hash(self[0]) ? ? l1 = MyList([1, 2]) ?# print(l1) -> [1, 2] d = {} d[l1] = l1 print(d) ?# {[1, 2]: [1, 2]} d[1] = 1 print(d) ?# {[1, 2]: [1, 2], 1: 1}
竟然沒有改成功而是新添加了一個(gè)鍵值對(duì),可self[0]就是1啊,哈希值一樣啊,怎么會(huì)不一樣呢?難道要鍵的值一樣才能判斷是同一個(gè)鍵嗎?重寫__eq__方法試一下。
class MyList(list): ? ? def __hash__(self): ? ? ? ? return hash(self[0]) ? ? ? def __eq__(self, other): ? ? ? ? return self[0] == other ? ? l1 = MyList([1, 2]) ?# print(l1) -> [1, 2] d = {} d[l1] = l1 print(d) ?# {[1, 2]: [1, 2]} d[1] = 1 print(d) ?# {[1, 2]: 1}
這回成功了,那就是__hash__返回值相等,且eq判斷也相等,才會(huì)被認(rèn)為是同一個(gè)鍵。那這兩個(gè)先判斷哪個(gè)呢?加個(gè)代碼試一下
class MyList(list): ? ? def __hash__(self): ? ? ? ? print('hash is run') ? ? ? ? return hash(self[0]) ? ? ? def __eq__(self, other): ? ? ? ? print('eq is run') ? ? ? ? return self[0] == other ? ? l1 = MyList([1, 2]) ?# print(l1) -> [1, 2] d = {} d[1] = 1 d[l1] = 'l1' print(d) ? ? # 結(jié)果: # hash is run # eq is run # {1: 'l1'}
__hash__先執(zhí)行,另外字典在內(nèi)存中存儲(chǔ)數(shù)據(jù)的位置和鍵的hash也是有關(guān)的,邏輯上也像印證。
先計(jì)算hash,找到相對(duì)應(yīng)的那片內(nèi)存空間,里面沒有值的話就直接寫入,對(duì)于字典來說就是新增鍵值對(duì);如果里面已經(jīng)有值了,那就判斷新來的鍵和原來的那里的鍵是不是相等,相等就認(rèn)為是一個(gè)鍵,對(duì)于字典來說就是更新值,不相等就再開空間,相當(dāng)于字典新增鍵值對(duì)。
在你驗(yàn)證自己想法的時(shí)候可能遇到__hash__和__eq__的一些想不到的麻煩,可以看這里:__hash__和__eq__的繼承使用問題
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Python機(jī)器學(xué)習(xí)之實(shí)現(xiàn)模糊照片人臉恢復(fù)清晰
GFPGAN是騰訊開源的人臉修復(fù)算法,它利用預(yù)先訓(xùn)練好的面部?GAN(如?StyleGAN2)中封裝的豐富和多樣的先驗(yàn)因素進(jìn)行盲臉?(blind?face)修復(fù)。這篇文章主要為大家介紹通過GFPGAN實(shí)現(xiàn)模糊照片人臉恢復(fù)清晰,需要的朋友可以參考一下2021-12-12Python+DeOldify實(shí)現(xiàn)老照片上色功能
DeOldify是一種技術(shù),以彩色和恢復(fù)舊的黑白圖像,甚至電影片段。它是由一個(gè)叫Jason?Antic的人開發(fā)和更新的。本文將利用DeOldify實(shí)現(xiàn)老照片上色功能,感興趣的可以了解一下2022-06-06Python創(chuàng)建多線程的兩種常用方法總結(jié)
這篇文章主要為大家詳細(xì)介紹了Python中創(chuàng)建多線程的兩種常用方法,文中的示例代碼簡(jiǎn)潔易懂,對(duì)我們掌握Python有一定的幫助,需要的可以收藏一下2023-05-05跟老齊學(xué)Python之復(fù)習(xí)if語句
是否記得,在上一部分,有一講專門介紹if語句的:從if開始語句的征程。在學(xué)習(xí)if語句的時(shí)候,對(duì)python編程的基礎(chǔ)知識(shí)了解的還不是很多,或許沒有做什么太復(fù)雜的東西。本講要對(duì)它進(jìn)行一番復(fù)習(xí),通過復(fù)習(xí)提高一下。如果此前有的東西忘記了,建議首先回頭看看前面那講。2014-10-10python之lambda表達(dá)式與sort函數(shù)中的key用法
這篇文章主要介紹了python之lambda表達(dá)式與sort函數(shù)中的key用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08python輸入整條數(shù)據(jù)分割存入數(shù)組的方法
今天小編就為大家分享一篇python輸入整條數(shù)據(jù)分割存入數(shù)組的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-11-11PyQt5按下按鍵選擇文件夾并顯示的實(shí)現(xiàn)
這篇文章主要介紹了PyQt5按下按鍵選擇文件夾并顯示的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-03-03Matplotlib繪圖基礎(chǔ)之配置參數(shù)詳解
Matplotlib?提供了大量配置參數(shù),這些參數(shù)可以但不限于讓我們從整體上調(diào)整通過?Matplotlib?繪制的圖形樣式,下面我們就來看看如何巧妙的運(yùn)用這些參數(shù)吧2023-08-08