本章介绍Python语言中的映射类型(字典)和集合类型,包括操作符、工厂函数、内建函数与方法。
字典是Python中唯一的映射类型——键key直接映射到值value。字典是容器类型,其对象是可变的。字典中的数据是无序排列的。是哈希表。
创建字典——直接赋值{}、工厂函数dict()、内建方法fromkeys():
>>> dict1={} >>> dict2={'name':'earth','port':80} >>> dict1, dict2 ({}, {'name': 'earth', 'port': 80}) >>> dict3=dict(['ab','cd']) >>> dict3 {'a': 'b', 'c': 'd'} >>> dict4={}.fromkeys(['ab','cd']) #对应同一值,默认对应None >>> dict5={}.fromkeys(('ab','cd'), -1) >>> dict4,dict5 ({'ab': None, 'cd': None}, {'ab': -1, 'cd': -1})
访问字典元素——dict[key]、dict.method():
>>> d={'a':1, 'b':2, 'c':3} >>> d {'a': 1, 'c': 3, 'b': 2} >>> d['b'] #直接通过dict[key]访问值
2
>>> d['d'] #如果key不存在则报错
Traceback (most recent call last): File "<stdin>", line 1, in <module> KeyError: 'd'
>>> 'a' in d #判断dict中是否有key
True >>> d.has_key('c') #用方法来判断是否有key,不再推荐使用
True >>> d.values(), d.keys(), d.items() #dict方法,分别返回 键、值、键值对 组成的无序列表
([1, 3, 2], ['a', 'c', 'b'], [('a', 1), ('c', 3), ('b', 2)]) >>> for key in d: #直接用迭代器访问字典,迭代的是字典的键
... print key ... a c b >>> print '%(a)d %(b)d %(c)d' % d #字典简化了print的格式化符 1 2 3
添加新元素/修改已存在元素——dict[key]=new_value:
>>> d['d'] = 4 #不存在的键,则添加键值对 >>> d['a'] = -1 #存在的键,则覆盖原来的值 >>> d {'a': -1, 'c': 3, 'b': 2, 'd': 4}
删除字典元素和字典:
1 del dict[key] #删除键为“name”的条目
2 dict.clear() #删除dict中所有的条目,成为空字典
3 del dict #删除整个dict字典
4 dict.pop(key) #删除并返回键为key的条目
比较运算符、逻辑运算符 同样适用于字典。
比较字典大小不常用,算法流程为:①比较长度,长度大的字典大。②长度一样时,比较键,键比较顺序和keys()方法返回的顺序一样。(相同的键会映射到哈希表的同一位置,保证了这一过程的一致性) ③键一样时,按照键的顺序比较相应的值。④之前的比较都一样,则两个字典相同。
键查找操作符[]、键成员操作符[not] in。
type()、str()、cmp()、len() 同样适用于字典。cmp()的比较过程见2.1。
dict()——参数可以是 空/可迭代的/另一个字典:
>>> dict() #生成空字典 {} >>> dict( zip(('x', 'y'), (1, 2)) ) #参数为一个列表,即一个可迭代对象,其中每个可迭代的元素必须成对出现 {'y': 2, 'x': 1} >>> dict({'x':1}) #参数为另一个字典,即浅拷贝,效率不高,不要使用这种方法 {'x': 1} >>> {'x':1}.copy() #高效率的浅拷贝 {'x': 1} >>> dict(ab=1,cd=2) #关键字参数 {'ab': 1, 'cd': 2}
hash(obj)——返回对象的哈希值,可用于判断是否能作为字典的键。如果不可哈希则产生TypeError错误。
1 dict.clear() #删除字典中所有元素
2 dict.copy() #返回字典(浅拷贝)的一个副本
3 dict.fromkeys(seq, val=None) #创建并返回一个新字典,以seq中的元素为键,val为所有键对应的初始值(默认为None)
4 dict.get(key, default=None) #读取key对应的值,如果不存在此键,则返回default的值(默认值为None)。
5 dict.has_key(key) #如果键存在,返回True。推荐使用成员操作符。
6 dict.items() #返回一个包含字典中(键, 值)对元组的列表
7 dict.iter*() #方法iteritems(),iterkeys(),itervalues(),返回一个迭代子而非列表,当元素很多时可以节省内存。
8 dict.keys() #返回一个包含字典中键的列表
9 dict.pop(key[, default]) #如果键存在,删除键值对并返回值;如果key键不存在,且没有指定default,则引发KeyError异常
10 dict.popitem() #删除并返回第一个键值对,如果字典为空则产生错误。
11 dict.setdefault(key, default=None) #如果不存在key键,由dict[key]=default为它赋值并返回值;存在则直接返回值
12 dict.update(dict2) #将字典dict2的键-值对添加到字典dict,重复的键则覆盖dict中原值
13 dict.values() #返回一个包含字典中所有值的列表
一个键只能对应一个值。
键必须是可哈希的,不可变类型——字符串、数字、只包含不可变类型的元组——都是可哈希的。值相同的数字哈希值也相同hash(1) == hash(1.0)。
实现了__hash__()特殊方法的类,如果该方法返回一个不可变类型,也是可哈希的。因为解释器调用哈希函数,根据字典中键的值来计算数据存储的位置。
集合对象是一组无序列的可哈希的值,集合成员可以用作字典的键。分为可变集合和不可变集合,可变集合不可哈希。
集合:支持[not] in检查成员、len()得到元素个数、for迭代集合成员。不支持索引、切片,没有键的概念。
Python中的集合和数学上的集合差不多,操作符如下:
1 数学符号 Python符号 说明 2 ∈ in 是集合成员 3 ∉ not in 不是集合成员 4 = == 等于 5 ≠ != 不等于 6 ⊂ < 真子集 7 ⊆ <= 子集 8 ⊃ > 严格超集 9 ⊇ >= 超集 10 ∩ & 交集 11 ∪ | 并集 12 - - 差补 13 △ ^ 对称差(A^B = A|B - A&B)
创建结合类型——只能用工厂函数set()和frozenset():
>>> s=set('abc') #可变集合,传入一个序列或者可迭代的对象。 >>> fs=frozenset([1,2,3]) #不可变集合 >>> s set(['a', 'c', 'b']) >>> fs frozenset([1, 2, 3])
访问集合成员——检查成员、遍历成员:
>>> 'c' in s #检查成员 True >>> for i in fs: #遍历成员 ... print i ... 1
2
3
更新集合——使用集合内建的方法和操作符,只能对可变集合,进行添加或删除元素(具体见第8节的内建函数):
>>> s.add(321) #可变集合添加元素 >>> s set(['a', 321, 'c', 'b']) >>> fs.remove(1) #不可变集合删除元素,错误 Traceback (most recent call last): File "<stdin>", line 1, in <module> AttributeError: 'frozenset' object has no attribute 'remove'
删除集合——del set即可。
成员操作符—— in、not in
集合等价—— ==、!=(两个集合的成员完全相同时,集合相同,不区分可变与不可变类型)
子集/超集—— 真子集<、子集<=、真超集>、超集>=
适用于所有集合——并集|、交集&、差补-、对称差^,返回一个新集合,参与运算的集合不变:
>>> s1=set([1,2]) >>> s2=set([2,3]) >>> fs=frozenset([2,3]) >>> s1, s2, fs (set([1, 2]), set([2, 3]), frozenset([2, 3])) >>> s1|s2, s1.union(s2) #s1和s2的交集,分别使用符号和方法 (set([1, 2, 3]), set([1, 2, 3])) >>> s1&s2, s1.intersection(s2) #交集 (set([2]), set([2])) >>> s1-s2, s1.difference(s2) #差补,即属于s1但是不属于s2的元素集合 (set([1]), set([1])) >>> s1^s2, s1.symmetric_difference(s2) #对称差,即s1|s2 - s1&s2 (set([1, 3]), set([1, 3])) >>> s1|fs, fs|s1 #可变与不可变类型集合运算,返回类型与左边的类型相同 (set([1, 2, 3]), frozenset([1, 2, 3]))
仅适用于可变集合——即集合运算的复合运算—— |=、&=、-=、^=,直接应用于左边的集合,没有返回值:
>>> s1 = set([1,2]) >>> s2 = set([1,2]) >>> s3 = set([2,3]) ###原始集合。以下运算都使用这三个数据,由于是在原集合上直接修改,每次运算都重新赋值,以下省略而已。
>>> s1 |= s3 #求并集并将新集合赋值给左值,没有返回值。分别使用 符号和方法。 >>> s2.update(s3) >>> s1,s2 (set([1, 2, 3]), set([1, 2, 3])) >>> s1 &= s3 #交集 >>> s2.intersection_update(s3) >>> s1,s2 (set([2]), set([2])) >>> s1 -= s3 #差补 >>> s2.difference_update(s3) >>> s1,s2 (set([1]), set([1])) >>> s1 ^= s3 #对称差 >>> s2.symmetric_difference_update(s3) >>> s1,s2 (set([1, 3]), set([1, 3]))
len()——返回集合成员个数。
set()、frozenset()——生成一个集合。参数必须是可迭代的——序列、迭代器、支持迭代的对象(文件或字典)。
适用于所有集合的,之前也讲述过:
1 s.issubset(t) #符号<= 如果s是t的子集,则返回True
2 s.issuperset(t) #符号>= 如果t是s的超集,则返回True
3 s.union(t) #符号| 返回一个新集合,该集合是s和t的并集
4 s.intersection(t) #符号& 返回一个新集合,该集合是s和t的交集
5 s.isdisjoint(t) #如果s和t没有交集,则返回True
6 s.difference(t) #符号- 返回一个新集合,该集合成员是s的成员,但不是t的成员
7 s.symmetric_difference(t) #符号^ 返回一个新集合,该集合是s或t的成员,但不是s和t共有的成员
8 s.copy() #返回一个新集合,它是集合s的浅拷贝。比工厂函数快
仅适用于可变集合,有之前讲述的,也有一些新方法:
1 s.update(t) #符号|= s为s和t的并集 2 s.intersection_update(t) #符号&= s为s和t的交集 3 s.difference_update(t) #符号-= s中的成员是属于s但不包含在t中的元素 4 s.symmetric_difference_update(t) #符号^= s中的成员是属于s或t,但不属于s和t共有 5 s.add(obj) #在集合s中添加对象obj 6 s.remove(obj) #从集合s中删除对象obj,如果obj不存在,则引发KeyError错误 7 s.discard(obj) #从集合s中删除对象obj,如果obj不存在也不报错 8 s.pop() #删除集合s中的任意一个对象,并返回它 9 s.clear() #删除集合s中的所有元素
操作符和方法:当用操作符时,操作符两边的操作数必须是集合。在使用内建方法时,对象也可以是迭代类型的。
7-8 人力资源。创建一个简单的雇员姓名和编号的程序,让用户输入一组雇员姓名和编号,你的程序可以提供按照姓名排序输出的功能,雇员姓名显示在前面,后面是对应的雇员编号。附加题:添加一项功能,按照雇员编号的顺序输出数据。
1 #!/usr/bin/env Python 2 3 '''sorted by key or value''' 4 5 idx = 0 6 emp = {} 7 8 def mycmp(a, b): 9 if a[idx] > b[idx]: 10 return 1 11 elif a[idx] < b[idx]: 12 return -1 13 else: 14 return 0 15 16 while True: 17 info = raw_input('Enter like this: name:id ,q to quit: ') 18 if info == 'q': 19 break 20 else: 21 info = info.split(':') 22 emp[info[0]] = int(info[1]) 23 24 print sorted(emp.items(), mycmp) 25 idx = 1 26 print sorted(emp.items(), mycmp)
7–13 随机数。使用random模块中的randint()或randrange()方法生成一个随机数集合:从0到9(包括9)中随机选择,生成1到10个随机数。这些数字组成集合A(A可以是可变集合,也可以不是)。同理,按此方法生成集合B。每次新生成集合A和B后,显示结果A|B和A&B。
1 #!/usr/bin/env Python 2 3 import random 4 5 def make_list(): 6 new_list = [] 7 for i in range(random.randint(1, 10)): 8 new_list.append(random.randint(0, 9)) 9 return set(new_list) 10 11 A = make_list() 12 B = make_list() 13 print A, B 14 15 print A|B 16 print A&B