Python 内置收据结构之九(字典)


一、字典 dict

  • key-value 键值对的数据的集合
  • 可变的无序的key 不重复(key 可哈希)

二、字典 dict 定义 初始化

  • d = dict() 或者 d = {}
  • dict(**kwargs) 使用 name = value 初始化一个字典
  • dict(iterable,**kwarg) 使用可迭代对象和 name = value 构造字典,不过可迭代对象的元素必须是一个 二元 结构
    • d = dict(((1,'a'), (2,'b')))d = dict(([1,'a'], [2,'b']))
  • dict(mapping, **kwarg) 使用一个字典构建另一个字典
  • d = {'a':10, 'b':20, 'c':None, 'd':[1,2,3]}
  • 类方法 dict.fromkeys(iterable, value=None)
    • d = dict.fromkeys(range(5))
    • d = dict.fromkeys(range(5),0)
d1 = {}
d2 = dict()
d3 = {1:1, '2':2, 'a':3, 'b':True, 'c':[1,2,3]}
d4 = {'a':100, 'a':'2'}   # 被后面的覆盖
d5 = dict([1,2], ['a',[300]], {'b', 400})   # 最后一组是集合,无序的,所以无法判断谁是 key
d6 = dict(a=1, b=2, c=3)
d7 = dict(d6)
d8 = dict.fromkeys(range(10))
l1 = [1]
d8 = dict.fromkeys(range(10), l1)   # 不推荐使用此方式,l1 引用变了,dict 的 value 会跟随变化
print(d8)
l1.append(2)
print(d8)
示例.png
示例2.png

三、字典元素的访问

3.1 d[key]
  • 返回 key 对应的值 value
  • key 不存在抛出 KeyError 异常
示例.png
3.2 get(key[, default])
  • 返回 key 对应的值 value
  • key 不存在返回缺省值,若没有设置缺省值就返回 None,不创建 kv 对
示例.png
3.3 setdefault(key[, default])
  • 返回 key 对应的值 value
  • key 不存在,添加 kv 对,value 设置为 default,并返回 default,若 default 没有设置,缺省为 None
示例.png

四、字典增加和修改

4.1 d[key] = value
  • key 对应的值修改为 value
  • key 不存在添加新的 kv 对
4.2 update([other]) -> None
  • 使用另一个字典的 kv 对更新本字典
  • key 不存在,就添加
  • key 存在,覆盖已经存在的 key 对应的值
  • 就地修改
d.update(red=1)
d.update((('red',2),))
d.update({'red':3})
示例.png

五、字典删除

5.1 pop(key[, default])
  • key 存在,移除它,并返回它的 value
  • key 不存在,返回给定的 default
  • default 为设置,key 不存在则抛出 KeyError 异常
示例.png
5.2 popitem()
  • 移除并返回一个任意的键值对
  • 字典为 empty,抛出 KeyError 异常
示例.png
5.3 clear()
  • 清空字典
5.4 del 语句
a = True
b = [6]
d = {'a':1, 'b':b, 'c':[1, 3, 5]}
del a   # 引用计数减一,True 为常量,并不能说没有引用
del d['c']   # 删除了一个对象 [1, 3, 5]?
del b[0]   # 列表清空,但列表本身还在,目前列表引用计数 2
c = b   # 列表引用计数 3
del c   # 引用计数 2
del b   # 引用计数 1
b = d['b']
  • del d['c'] 看着像删除了一个对象,本质上减少了一个对象的引用,del 实际上删除的是名称,而不是对象

六、字典的遍历和移除

6.1 for ... in dict
  • 遍历 key
for k in d:
    print(k)
示例.png
for k in d.keys():
    print(k)
示例.png
for v in d.values():
    print(v)
示例.png
  • 遍历 item,即 kv 对
for item in d.items():
    print(item)
示例.png
for item in d.items():
    print(item[0], item[1])
示例.png
for k,v in d.items():
    print(k, v)
示例.png
for k,_ in d.items():
    print(k)

for _,v in d.items():
    print(v)
6.2 总结
  • Python 3 中,keysvaluesitems 方法返回一个类似生成器的可迭代对象,不会把函数的返回结果复制到内存中

    • 返回 Dictionary view 对象,可使用 len()iter()in 操作
    • 字典的 entry 的动态的视图,字典变化,视图将反映出这些变化
    • keys 返回一个类 set 对象,也就是可看作一个 set 集合
    • values 都可 hash,那么 items 也可看作是类 set 对象
  • Python 2 中,上面的方法会返回一个新的列表,占据新的内存空间,所以 Python 2 建议使用 iterkeysitervaluesitertiems 版本,返回一个迭代器,而不是返回一个 copy

6.3 如何在遍历的时候移除元素
  • 错误的做法
d = dict(a=1, b=2, c='abc')
for k,v in d.items():
    d.pop(k)   # 抛异常

while len(d):   # 相当于清空,不如直接 clear()
    print(d.popitem())

while d:
    print(d.popitem())
示例.png
示例.png
示例.png
示例.png
  • 正确的做法
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)
示例.png

七、字典的 key

7.1 key 的要求的 set 的元素要求一致
  • set 的元素可看做 keyset 可看做 dict 简化版
  • hashable 可哈希才可作为 key,可使用 hash() 测试
  • d = {1:0, 2.0:3, "abc":None, ('hello','world','python'):"string", b'abc':'135'}

八、defaultdict

8.1 collections.defaultdict([default_factory[, ...]])
  • 第一个参数是 default_factory,缺省是 None,它提供一个初始化函数,key 不存在的时候,会调用这个工厂函数来生成 key 对应的 value
  • 构造一个字典,values 是列表,为其添加随机个元素
import random

d1 = {}
for k in 'abcdef':
    for v in range(random.randint (1, 5)):
        if k not in d1.keys():
            d1[k] = []
        d1[k].append(v)
print(d1)
示例.png
from collections import defaultdict
import random

d2 = defaultdict(list)
for k in 'dbcdef':
    for v in range(random.randint(1, 5)):
        d2[k].append(v)
print(d2)
示例.png

九、OrderedDict

9.1 collections.OrderedDict([items])
  • key 并不是按照加入的顺序排列,可使用 OrderedDict 记录顺序
from collections import OrderedDict
import random

d = {'banana': 3, 'apple': 4, '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())
示例.png
  • 有序字典可记录元素插入的顺序,打印的时候也是按照这个顺序输出打印
  • 3.6 版本的 Python 字典就是记录 key 插入的顺序 (IPython 不一定有效果)
9.2 应用场景

假如使用字典记录了 N 个产品,这些产品使用 ID 由小到大加入到字典中
除了使用字典检索的遍历,有时候需要取出 ID,但希望是按照输入顺序,因为输入顺序是有序的
否则还要重新把遍历到的值排序

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容