字典

字典

key-value鍵值對的數據的集合,字典是可變的無序的、同時key不重複。字典(dict)python中惟一的一個映射類型.他是以{ }括起來的鍵值對組成. dictkey是惟一的在保存的時候根據key來計算出一個內存地址而後將key-value保存在這個地址中。這種算法被稱爲hash算法因此切記dict中存儲的key-value中的key'必須是可hash的。

已知的可哈希(不可變)的數據類型: int, str, tuple, bool
不可哈希(可變)的數據類型: list, dict, set

python

定義和初始化

使用{}或者dict()定義字典。算法

d = dict()
print(type(d))
e = {}
type(e)

結果爲:
<class 'dict'>
dict

# 不合法
# dic = {[1, 2, 3]: '周杰倫'} # list是可變的. 不能做爲key
# dic = {{1: 2}: "哈哈哈"} # dict是可變的. 不能做爲key
dic = {{1, 2, 3}: '呵呵呵'} # set是可變的, 不能做爲key

dict(**kwargs) 使用name=value對初始化一個字典,dict(iterable, **kwarg) 使用可迭代對象和name=value對構造字典,不過可迭代對象的元素必須是一個二元結構。
app

d = dict(((1,'a'),(2,'b')))
print(d)

結果爲:
{1: 'a', 2: 'b'}

d = dict(([1,'a'],[2,'b']))
print(d)

結果爲:
{1: 'a', 2: 'b'}

e = enumerate(range(4))
d = dict(e)
print(d)dom

 

結果爲:函數

{0: 0, 1: 1, 2: 2, 3: 3}

dict(mapping, **kwarg) 使用一個字典構建另外一個字典。測試

d = {'a':10, 'b':20, 'c':None, 'd':[1,2,3]}

結果爲:
{'a': 10, 'b': 20, 'c': None, 'd': [1, 2, 3]}

類方法dict.fromkeys(iterable, value)
spa

d = dict.fromkeys(range(5))
print(d)

結果爲:
{0: None, 1: None, 2: None, 3: None, 4: None}

d = dict.fromkeys(range(5),0)
print(d)

結果爲:
{0: 0, 1: 0, 2: 0, 3: 0, 4: 0}

#應該特別注意值的引用。
d = dict.fromkeys(range(5),[1,2])
print(d)
d[4].append(3)
print(d)code

 

結果爲:對象

{0: [1, 2], 1: [1, 2], 2: [1, 2], 3: [1, 2], 4: [1, 2]}
{0: [1, 2, 3], 1: [1, 2, 3], 2: [1, 2, 3], 3: [1, 2, 3], 4: [1, 2, 3]}

字典元素的訪問

d[key],返回key對應的值value,key不存在拋出KeyError異常。
blog

d = {"a":1,"b":2,"c":3}
print(d["b"])
print(d["d"])

結果爲:

2
---------------------------------------------------------------------------
KeyError                                  Traceback (most recent call last)
<ipython-input-24-98a066a8e141> in <module>
      1 d = {"a":1,"b":2,"c":3}
      2 print(d["b"])
----> 3 print(d["d"])

KeyError: 'd'

get(key[, default]),返回key對應的值value,key不存在返回缺省值,若是沒有設置缺省值就返回None。

dic = {"id": 123, "name": 'xpc', "age": 18}
print(dic.get("id",456))
print(dic.get("idd",456))
print(dic.get("idd"))

結果爲:
123
456
None

setdefault(key[, default]) ,返回key對應的值value,key不存在,添加kv對,value爲default,並返回default,若是default沒有設置,缺省爲None 。若是dict中已經存在了. 那麼setdefault將不會起做用。

dic = {"id": 123, "name": 'xpc', "age": 18}
print(dic.setdefault("id"))
print(dic.setdefault("sex","male"),dic)
print(dic.setdefault("idd"),dic)
dic.setdefault("id",123)
print(dic)

結果爲:

123
male {'id': 123, 'name': 'xpc', 'age': 18, 'sex': 'male'}
None {'id': 123, 'name': 'xpc', 'age': 18, 'sex': 'male', 'idd': None}
{'id': 123, 'name': 'xpc', 'age': 18, 'sex': 'male', 'idd': None}

字典的增長和修改

d[key] = value,將key對應的值修改成value,key不存在添加新的kv對。

d = {"a":1,"b":2,"c":3}
d["a"] = 10
d["d"] = 5
print(d)

結果爲:
{'a': 10, 'b': 2, 'c': 3, 'd': 5}

update([other]) -> None,使用另外一個字典的kv對更新本字典,key不存在,就添加,key存在,覆蓋已經存在的key對應的值,它是就地修改。

c1 = {}
c1.update(red=1)
print(c1)

結果爲:

{'red': 1}

c1.update((('red',2),))
print(c1)

結果爲:
{'red': 2}

d1 = {"green":1}
d1.update({'red':3})
print(d1)

結果爲:
{'green': 1, 'red': 3}
dic = {"id": 123, "name": 'xpc', "age": 18}
dic1 = {"id": 456, "name": "xpcs", "ok": "wtf"}
dic.update(dic1) # 把dic1中的內容更新到dic中. 若是key重名. 則修改替換. 若是不存在key, 則新增.
print(dic)
print(dic1)

結果爲:

{'id': 456, 'name': 'xpcs', 'age': 18, 'ok': 'wtf'}
{'id': 456, 'name': 'xpcs', 'ok': 'wtf'}

字典刪除

pop(key[, default]),key存在,移除它,並返回它的value,key不存在,返回給定的default,default未設置,key不存在則拋出KeyError異常。

dic = {"id": 123, "name": 'xpc', "age": 18}
ret = dic.pop("id")
print(ret)

print(dic.pop("idd",-1))
print(dic.pop("idd"))

結果爲:

123
-1
---------------------------------------------------------------------------
KeyError                                  Traceback (most recent call last)
<ipython-input-53-57358572fa03> in <module>
      4 
      5 print(dic.pop("idd",-1))
----> 6 print(dic.pop("idd"))

KeyError: 'idd'

popitem(),移除並返回一個任意的鍵值對,字典爲empty,拋出KeyError異常。

dic = {"id": 123, "name": 'xpc', "age": 18}
ret = dic.popitem()
print(ret)

結果爲:
('age', 18)


dic = {}
print(dic.popitem())

結果爲:

KeyError                                  Traceback (most recent call last)
<ipython-input-57-86a9c7ea7a09> in <module>
      1 dic = {}
----> 2 print(dic.popitem())

KeyError: 'popitem(): dictionary is empty'


dic = {"id": 123, "name": 'xpc', "age": 18}
a,b = dic.popitem()
print(a,b)

結果爲:

age 18

clear(),清空字典。

dic = {"id": 123, "name": 'xpc', "age": 18}
dic.clear()
print(dic)

結果爲:

{}

del 語句

a  = True
b = [6]
d = {"a":1,"b":b,"c":[1,3,5]}
del a 
a

結果爲:

---------------------------------------------------------------------------
NameError                                 Traceback (most recent call last)
<ipython-input-15-23612501a197> in <module>
      3 d = {"a":1,"b":b,"c":[1,3,5]}
      4 del a
----> 5 a

NameError: name 'a' is not defined


del d["c"]
print(d)

結果爲:
{'a': 1, 'b': [6]}

del b[0]
print(b,d)

結果爲:
[] {'a': 1, 'b': []}

c = b
print(b,c)
print(d)

結果爲:
[] []
{'a': 1, 'b': []}

del c 
print(b,d)
print(c)

結果爲:
[] {'a': 1, 'b': []}
---------------------------------------------------------------------------
NameError                                 Traceback (most recent call last)
<ipython-input-21-c15658afec2c> in <module>
      1 del c
      2 print(b,d)
----> 3 print(c)

NameError: name 'c' is not defined


del b 
print(d)
print(b)

結果爲:
{'a': 1, 'b': []}
---------------------------------------------------------------------------
NameError                                 Traceback (most recent call last)
<ipython-input-22-5aac62a9d7d7> in <module>
      1 del b
      2 print(d)
----> 3 print(b)

NameError: name 'b' is not defined

b = d["b"]
print(b,d)

結果爲:
[] {'a': 1, 'b': []}

由以上例子可知道,del a['c'] 看着像刪除了一個對象,本質上減小了一個對象的引用,del 實際上刪除的是名稱,而不是對象。

字典的遍歷

遍歷key,有兩種方法,一種是for ……in dic,另外一種是for……in dic.keys()

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic:
    print(a)

結果爲:

id
name
age


dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic.keys():
    print(a)

結果爲:

id
name
age

遍歷value,有三種方法,for k in d:print(d[k]) ,for k in d.keys():print(d.get(k)) ,for v in d.values():print(v) 

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic:
    print(dic[a])

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic:
    print(dic.get(a))

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic.keys():
    print(dic[a])

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic.keys():
    print(dic.get(a))

dic = {"id": 123, "name": 'xpc', "age": 18}
for a in dic.values():
    print(a)

結果都爲:

123
xpc
18

遍歷item,即kv對。

dic = {"id": 123, "name": 'xpc', "age": 18}
for item in dic.items():
    print(item)

結果爲:
('id', 123)
('name', 'xpc')
('age', 18)

dic = {"id": 123, "name": 'xpc', "age": 18}
for item in dic.items():
    print(item[0], item[1])

結果爲:

id 123
name xpc
age 18

dic = {"id": 123, "name": 'xpc', "age": 18}
for k,v in dic.items():
    print(k, v)

結果爲:
id 123
name xpc
age 18

dic = {"id": 123, "name": 'xpc', "age": 18}
for k, _ in dic.items():
    print(k)

結果爲:

id
name
age

dic = {"id": 123, "name": 'xpc', "age": 18}
for _ ,v in dic.items():
    print(v)

結果爲:
123
xpc
18

Python3中,keys、values、items方法返回一個相似一個生成器的可迭代對象,不會把函數的返回結果複製到內存中,Dictionary view對象,字典的entry的動態的視圖,字典變化,視圖將反映出這些變化。
Python2中,上面的方法會返回一個新的列表,佔據新的內存空間。因此Python2建議使用iterkeys、itervalues、iteritems版本,返回一個迭代器,而不是一個copy。(不懂)

字典的遍歷和移除

如何在遍歷的時候移除元素,錯誤的作法是直接在遍歷中pop。

#錯誤作法
d = dict(a=1, b=2, c='abc')
print(d)
for k,v in d.items():
    d.pop(k) # 異常

{'a': 1, 'b': 2, 'c': 'abc'}
---------------------------------------------------------------------------
RuntimeError                              Traceback (most recent call last)
<ipython-input-105-faeb0e8349bb> in <module>
      1 d = dict(a=1, b=2, c='abc')
      2 print(d)
----> 3 for k,v in d.items():
      4     d.pop(k) # 異常

RuntimeError: dictionary changed size during iteration


d = dict(a=1, b=2, c='abc')
while len(d): # 至關於清空,不如直接clear()
    print(d.popitem())

結果爲:
('c', 'abc')
('b', 2)
('a', 1)

#正確的作法
d = dict(a=1, b=2, c='abc')
keys = []
for k,v in d.items():
    if isinstance(v, str):
        keys.append(k)
        
for k in keys:
    d.pop(k)
print(d)

結果爲:
{'a': 1, 'b': 2}

字典的key

key的要求和set的元素要求一致,set的元素能夠就是看作key,set能夠看作dict的簡化版 ,hashable 可哈希才能夠做爲key,可使用hash()測試 .

d = {1 : 0, 2.0 : 3, "abc" : None, ('hello', 'world', 'python') : "string", b'abc' : '135'}
print(d)

結果爲:

{1: 0, 2.0: 3, 'abc': None, ('hello', 'world', 'python'): 'string', b'abc': '135'}

defaultdict

collections.defaultdict([default_factory[, ...]]) ,一個參數是default_factory,缺省是None,它提供一個初始化函數。當key不存在的時候,會調用這個工廠函數來生成key對應的value。

import random
d1 = {}
for k in 'abcdef':
    for i in range(random.randint(1,5)):
        if k not in d1.keys():
            d1[k] = []
        d1[k].append(i)
print(d1)

結果爲:

{'a': [0], 'b': [0, 1, 2], 'c': [0, 1, 2, 3, 4], 'd': [0, 1], 'e': [0, 1, 2], 'f': [0]}

from collections import defaultdict
import random
d1 = defaultdict(list)
for k in 'abcdef':
    for i in range(random.randint(1,5)):
        d1[k].append(i)
print(d1)

結果爲:
defaultdict(<class 'list'>, {'a': [0, 1, 2, 3, 4], 'b': [0, 1, 2, 3, 4], 'c': [0, 1, 2], 'd': [0, 1, 2], 'e': [0], 'f': [0, 1]})

OrderedDict

collections.OrderedDict([items]) ,key並非按照加入的順序排列,可使用OrderedDict記錄順序

from collections import OrderedDict
import random
d = {'banana': 3, 'apple': 4, 'pear': 1, 'orange': 2}
print(d)
keys = list(d.keys())
random.shuffle(keys)
print(keys)
od = OrderedDict()
for key in keys:
    od[key] = d[key]
print(od)
print(od.keys())

結果爲:

{'banana': 3, 'apple': 4, 'pear': 1, 'orange': 2}
['pear', 'apple', 'banana', 'orange']
OrderedDict([('pear', 1), ('apple', 4), ('banana', 3), ('orange', 2)])
odict_keys(['pear', 'apple', 'banana', 'orange'])

有序字典能夠記錄元素插入的順序,打印的時候也是按照這個順序輸出打印,3.6版本的Python的字典就是記錄key插入的順序(IPython不必定有效果)。

應用場景:

  • 假如使用字典記錄了N個產品,這些產品使用ID由小到大加入到字典中
  • 除了使用字典檢索的遍歷,有時候須要取出ID,可是但願是按照輸入的順序,由於輸入順序是有序的
  • 不然還須要從新把遍歷到的值排序 

練習1:用戶輸入一個數字,打印每個數字及其重複的次數。

num = input("請輸入一個數字:")
d = {}
for c in num:
    if c not in d.keys():
        d[c]=1
    else:
        d[c]+=1
    
print(d)

num = input("請輸入一個數字:")
d = {}
for c in num:
    if not d.get(c):
        d[c]=1
        continue
    d[c]+=1
print(d)

結果爲:

請輸入一個數字:12343
{'1': 1, '2': 1, '3': 2, '4': 1}

數字重複統計:隨機產生100個整數,數字的範圍是【-1000,1000】,升序輸出數字及其重複的次數。

import random
n = 100
nums = [0]*n
for i in range(n):
    nums[i] = random.randint(-1000,1000)
print(nums)
t=nums.copy()
t.sort()
print(t)

d = {}
for x in nums: 
    if x not in d.keys():
        d[x]=1
    else:
        d[x]+=1
print(d)
d1 = sorted(d.items())
print(d1)
   

結果爲:
[78, -325, 533, 125, 742, 0, 886, 31, -135, 775, -191, 798, -426, 415, -662, 958, -16, -159, 223, -163, 52, -455, -903, 584, -647, 152, -790, 150, -653, -811, 498, -462, -518, -512, 787, 613, -362, 510, 982, 239, -97, 326, 318, 624, 20, 793, -811, 149, -501, -18, -548, -444, -195, 722, 657, 798, -850, 398, 151, -91, -581, 763, -986, 185, -57, 103, -554, 707, 964, -675, 408, 207, -382, -312, -794, -857, -139, -293, 792, -987, 976, 127, -291, -906, 220, -733, -542, 701, -680, -981, 288, -868, -283, 432, -250, -916, 908, -625, 849, 872]
[-987, -986, -981, -916, -906, -903, -868, -857, -850, -811, -811, -794, -790, -733, -680, -675, -662, -653, -647, -625, -581, -554, -548, -542, -518, -512, -501, -462, -455, -444, -426, -382, -362, -325, -312, -293, -291, -283, -250, -195, -191, -163, -159, -139, -135, -97, -91, -57, -18, -16, 0, 20, 31, 52, 78, 103, 125, 127, 149, 150, 151, 152, 185, 207, 220, 223, 239, 288, 318, 326, 398, 408, 415, 432, 498, 510, 533, 584, 613, 624, 657, 701, 707, 722, 742, 763, 775, 787, 792, 793, 798, 798, 849, 872, 886, 908, 958, 964, 976, 982]
{78: 1, -325: 1, 533: 1, 125: 1, 742: 1, 0: 1, 886: 1, 31: 1, -135: 1, 775: 1, -191: 1, 798: 2, -426: 1, 415: 1, -662: 1, 958: 1, -16: 1, -159: 1, 223: 1, -163: 1, 52: 1, -455: 1, -903: 1, 584: 1, -647: 1, 152: 1, -790: 1, 150: 1, -653: 1, -811: 2, 498: 1, -462: 1, -518: 1, -512: 1, 787: 1, 613: 1, -362: 1, 510: 1, 982: 1, 239: 1, -97: 1, 326: 1, 318: 1, 624: 1, 20: 1, 793: 1, 149: 1, -501: 1, -18: 1, -548: 1, -444: 1, -195: 1, 722: 1, 657: 1, -850: 1, 398: 1, 151: 1, -91: 1, -581: 1, 763: 1, -986: 1, 185: 1, -57: 1, 103: 1, -554: 1, 707: 1, 964: 1, -675: 1, 408: 1, 207: 1, -382: 1, -312: 1, -794: 1, -857: 1, -139: 1, -293: 1, 792: 1, -987: 1, 976: 1, 127: 1, -291: 1, -906: 1, 220: 1, -733: 1, -542: 1, 701: 1, -680: 1, -981: 1, 288: 1, -868: 1, -283: 1, 432: 1, -250: 1, -916: 1, 908: 1, -625: 1, 849: 1, 872: 1}
[(-987, 1), (-986, 1), (-981, 1), (-916, 1), (-906, 1), (-903, 1), (-868, 1), (-857, 1), (-850, 1), (-811, 2), (-794, 1), (-790, 1), (-733, 1), (-680, 1), (-675, 1), (-662, 1), (-653, 1), (-647, 1), (-625, 1), (-581, 1), (-554, 1), (-548, 1), (-542, 1), (-518, 1), (-512, 1), (-501, 1), (-462, 1), (-455, 1), (-444, 1), (-426, 1), (-382, 1), (-362, 1), (-325, 1), (-312, 1), (-293, 1), (-291, 1), (-283, 1), (-250, 1), (-195, 1), (-191, 1), (-163, 1), (-159, 1), (-139, 1), (-135, 1), (-97, 1), (-91, 1), (-57, 1), (-18, 1), (-16, 1), (0, 1), (20, 1), (31, 1), (52, 1), (78, 1), (103, 1), (125, 1), (127, 1), (149, 1), (150, 1), (151, 1), (152, 1), (185, 1), (207, 1), (220, 1), (223, 1), (239, 1), (288, 1), (318, 1), (326, 1), (398, 1), (408, 1), (415, 1), (432, 1), (498, 1), (510, 1), (533, 1), (584, 1), (613, 1), (624, 1), (657, 1), (701, 1), (707, 1), (722, 1), (742, 1), (763, 1), (775, 1), (787, 1), (792, 1), (793, 1), (798, 2), (849, 1), (872, 1), (886, 1), (908, 1), (958, 1), (964, 1), (976, 1), (982, 1)]

字符串重複統計:字符表「abcdefghijklmnopqrstuvwsyz」,隨機挑選2個字母組成字符串,共挑選100個,降序輸出這100個字符串及重複的次數。

import random

alphabet = "abcdefghijklmnopqrstuvwsyz" 
words = []
for _ in range(100):
    #words.append("".join(random.choice(alphabet) for _ in range(2) ))#生成器
    #words.append(random.choice(alphabet)+random.choice(alphabet))
    words.append("".join(random.sample(alphabet,2)))#隨即採樣
    
d = {}
for x in words:
    d[x]= d.get(x,0)+1
print(d)

d1 = sorted(d.items(),reverse=True)
print(d1)

結果爲:

{'gz': 1, 'do': 1, 'rl': 1, 'bg': 1, 'wo': 2, 'hu': 1, 'sf': 2, 'uh': 1, 'ol': 1, 'ag': 1, 'gc': 2, 'dw': 1, 'to': 1, 'cp': 1, 'sq': 1, 'wp': 1, 'mh': 1, 'ot': 2, 'pw': 1, 'oa': 1, 'qm': 1, 'py': 2, 'fe': 1, 'nt': 1, 'jy': 1, 'of': 1, 'ca': 1, 'tf': 1, 'kw': 1, 'qs': 2, 'jq': 1, 'he': 1, 'ep': 1, 'qe': 1, 'em': 1, 'ua': 1, 'zw': 1, 'ye': 1, 've': 1, 'rw': 1, 'pr': 1, 'mz': 1, 'ov': 1, 'aw': 1, 'rm': 1, 'qg': 1, 'pj': 1, 'za': 2, 'hr': 1, 'si': 1, 'lc': 1, 'uw': 1, 'qa': 2, 'pn': 1, 'ue': 1, 'wg': 1, 'ae': 1, 'cy': 1, 'hp': 1, 'qu': 1, 'fy': 1, 'ds': 1, 'yu': 1, 'ek': 1, 'sn': 1, 'op': 1, 'eb': 1, 'aq': 2, 'jf': 1, 'qp': 1, 'rs': 1, 'ko': 1, 'th': 1, 'fs': 1, 'qi': 1, 'pq': 1, 'yg': 1, 'om': 1, 'ew': 1, 'lf': 1, 'ka': 1, 'zk': 1, 'bs': 1, 'fu': 1, 'ec': 1, 'zc': 1, 'gs': 1, 'lz': 1, 'qv': 1, 'fr': 1, 'oy': 1}
[('zw', 1), ('zk', 1), ('zc', 1), ('za', 2), ('yu', 1), ('yg', 1), ('ye', 1), ('wp', 1), ('wo', 2), ('wg', 1), ('ve', 1), ('uw', 1), ('uh', 1), ('ue', 1), ('ua', 1), ('to', 1), ('th', 1), ('tf', 1), ('sq', 1), ('sn', 1), ('si', 1), ('sf', 2), ('rw', 1), ('rs', 1), ('rm', 1), ('rl', 1), ('qv', 1), ('qu', 1), ('qs', 2), ('qp', 1), ('qm', 1), ('qi', 1), ('qg', 1), ('qe', 1), ('qa', 2), ('py', 2), ('pw', 1), ('pr', 1), ('pq', 1), ('pn', 1), ('pj', 1), ('oy', 1), ('ov', 1), ('ot', 2), ('op', 1), ('om', 1), ('ol', 1), ('of', 1), ('oa', 1), ('nt', 1), ('mz', 1), ('mh', 1), ('lz', 1), ('lf', 1), ('lc', 1), ('kw', 1), ('ko', 1), ('ka', 1), ('jy', 1), ('jq', 1), ('jf', 1), ('hu', 1), ('hr', 1), ('hp', 1), ('he', 1), ('gz', 1), ('gs', 1), ('gc', 2), ('fy', 1), ('fu', 1), ('fs', 1), ('fr', 1), ('fe', 1), ('ew', 1), ('ep', 1), ('em', 1), ('ek', 1), ('ec', 1), ('eb', 1), ('dw', 1), ('ds', 1), ('do', 1), ('cy', 1), ('cp', 1), ('ca', 1), ('bs', 1), ('bg', 1), ('aw', 1), ('aq', 2), ('ag', 1), ('ae', 1)]
相關文章
相關標籤/搜索
本站公眾號
   歡迎關注本站公眾號,獲取更多信息