key-value鍵值對的數據的集合,字典是可變的、無序的、同時key不重複。字典(dict)是python中惟一的一個映射類型.他是以{ }括起來的鍵值對組成. 在dict中key是惟一的. 在保存的時候, 根據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的要求和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'}
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]})
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不必定有效果)。
應用場景:
練習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)]