欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

解讀Python中字典的key都可以是什么

 更新時(shí)間:2022年09月27日 16:40:52   作者:Inotime  
這篇文章主要介紹了解讀Python中字典的key都可以是什么,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

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ù)清晰

    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-12
  • python類裝飾器用法實(shí)例

    python類裝飾器用法實(shí)例

    這篇文章主要介紹了python類裝飾器用法,實(shí)例分析了Python類裝飾器的相關(guān)使用方法,需要的朋友可以參考下
    2015-06-06
  • Python+DeOldify實(shí)現(xiàn)老照片上色功能

    Python+DeOldify實(shí)現(xiàn)老照片上色功能

    DeOldify是一種技術(shù),以彩色和恢復(fù)舊的黑白圖像,甚至電影片段。它是由一個(gè)叫Jason?Antic的人開發(fā)和更新的。本文將利用DeOldify實(shí)現(xiàn)老照片上色功能,感興趣的可以了解一下
    2022-06-06
  • Python創(chuàng)建多線程的兩種常用方法總結(jié)

    Python創(chuàng)建多線程的兩種常用方法總結(jié)

    這篇文章主要為大家詳細(xì)介紹了Python中創(chuàng)建多線程的兩種常用方法,文中的示例代碼簡(jiǎn)潔易懂,對(duì)我們掌握Python有一定的幫助,需要的可以收藏一下
    2023-05-05
  • 跟老齊學(xué)Python之復(fù)習(xí)if語句

    跟老齊學(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-10
  • python之lambda表達(dá)式與sort函數(shù)中的key用法

    python之lambda表達(dá)式與sort函數(shù)中的key用法

    這篇文章主要介紹了python之lambda表達(dá)式與sort函數(shù)中的key用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • python輸入整條數(shù)據(jù)分割存入數(shù)組的方法

    python輸入整條數(shù)據(jù)分割存入數(shù)組的方法

    今天小編就為大家分享一篇python輸入整條數(shù)據(jù)分割存入數(shù)組的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2018-11-11
  • python2與python3共存問題的解決方法

    python2與python3共存問題的解決方法

    這篇文章主要為大家詳細(xì)介紹了python2與python3共存問題的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-09-09
  • PyQt5按下按鍵選擇文件夾并顯示的實(shí)現(xiàn)

    PyQt5按下按鍵選擇文件夾并顯示的實(shí)現(xiàn)

    這篇文章主要介紹了PyQt5按下按鍵選擇文件夾并顯示的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-03-03
  • Matplotlib繪圖基礎(chǔ)之配置參數(shù)詳解

    Matplotlib繪圖基礎(chǔ)之配置參數(shù)詳解

    Matplotlib?提供了大量配置參數(shù),這些參數(shù)可以但不限于讓我們從整體上調(diào)整通過?Matplotlib?繪制的圖形樣式,下面我們就來看看如何巧妙的運(yùn)用這些參數(shù)吧
    2023-08-08

最新評(píng)論