文章目录
一、字典
- 字典是‘键值对’的无序可变序列,字典中的每个元素都是一个“键值对”,包含“键对象”和“值对象”。
一个典型的字典定义方式:
a={‘name’:‘Zfloss’,‘age’:‘18’,‘job’,‘programmer’}
1. 字典的创建
- 通过{ }、dict()来创建字典对象
- 通过zip()创建字典对
k=[‘name’,‘age’,‘job’]
v=[‘Zfloss’,‘23’,‘programmer’]
d=dict(zip(k,v))
d={‘name’:‘Zfloss’,‘age’:‘23’,‘job’:‘programmer’}
- 通过fromkeys创建值为空的字典
a=dict.fromkeys(['name','age','job'])
print(a)
运行结果:
{['name':None,'age':None,'job':None}
2. 字典元素的访问
a={‘name’:‘Zfloss’,‘age’:‘18’,‘job’,‘programmer’}
- 通过键获得值,若键不存在,抛出异常
8a={'name':'Zfloss','age':'18','job':'programmer'}
print(a['name'])
print(a['age'])
print(a['job'])
print(a[sex])
运行结果:
Zfloss
18
programmer
抛出异常
- 通过get()方法获得“值”。推荐使用。优点是:指定键不存在,返回None;也可以设定指定键不存在是默认返回的对象。推荐使用get()获取“值对象”。
a.get('name') #字典中已经有的就直接输入‘键’得到‘值’
'Zfloss'
a.get('sex','不存在') #默认是None
'不存在'
a.get('sex','男人') #字典中没有的,也可以指定一个
男人
- 列出所有的键值对,列出所有的键,列出所有的值
a={'name':'Zfloss','age':'18','job':'programmer'}
print(a.values()) #列出所有值
print(a.keys()) #列出所欲键
print(a.items()) #列出所有键值对
运行结果:
dict_values(['Zfloss', '18', 'programmer'])
dict_keys(['name', 'age', 'job'])
dict_items([('name', 'Zfloss'), ('age', '18'), ('job', 'programmer')])
- len()键值对的个数
- 检测一个键是否在字典中
print('name' in a)
运行结果:
Ture
3. 字典元素添加、修改、删除
- 添加键值对,原字典有的键值对就是更新,没有的就是添加
{'name': 'Zfloss', 'age': '18', 'job': 'programmer', 'address': 'China'}
a['address']='China'
print(a)
运行结果:
- 使用updata()将新字典中所有键值对全部添加到旧字典对象上,如果key有重复,则直接覆盖。
a={'name':'Zfloss','age':'18','job':'programmer'}
b={'age':'23','color':'red'}
a.update(b)
print(a)
运行结果:
{'name': 'Zfloss', 'age': '23', 'job': 'programmer', 'color': 'red'}
- 字典中元素的删除,可以使用del()方法;或者clear()删除所有键值对;pop()删除指定键值对,并返回对应的’值对象’;
a={'name':'Zfloss','age':'18','job':'programmer'}
del(a['name'])
print(a)
b=a.pop('age')
print(a)
print(b)
a['name']='Zfloss'
a.clear()
print(a)
运行结果:
{'age': '18', 'job': 'programmer'}
{'job': 'programmer'}
18
{}
- popitem():随机删除和返回该键值对。
a={'name':'Zfloss','age':'18','job':'programmer'}
print(a.popitem())
print(a)
print(a.popitem())
print(a)
运行结果:
('job', 'programmer')
{'name': 'Zfloss', 'age': '18'}
('age', '18')
{'name': 'Zfloss'}
4. 序列解包
- 序列解包可以用于元组,列表,字典。序列解包可以让我们方便的对多个变量赋值。
x={'name':'Zfloss','age':'18','job':'programmer'}
a,b,c=x.keys()
print(a,end=' ')
print(b,end=' ')
print(c,)
d,e,f=x.values()
print(d,end=' ')
print(e,end=' ')
print(f,)
g,h,l=x.items()
print(g,end=' ')
print(h,end=' ')
print(l)
运行结果:
name age job
Zfloss 18 programmer
('name', 'Zfloss') ('age', '18') ('job', 'programmer')
5. 字典数据使用和列表存储,并实现访问
r1={'name':'赵一','age':26,'salary':30000,'city':'杭州'}
r2={'name':'赵二','age':27,'salary':40000,'city':'北京'}
r3={'name':'赵三','age':28,'salary':50000,'city':'深圳'}
tb=[r1,r2,r3]
for m in range(len(tb)):
print(tb[m].get('name'),tb[m].get('age'),tb[m].get('salary'),tb[m].get('city'))
运行结果:
赵一 26 30000 杭州
赵二 27 40000 北京
赵三 28 50000 深圳
6.字典核心底层原理(重要)
- 字典对象的核心是散列表。散列表是一个稀疏数组(总是有空白元素的数组),数组的 每个单元叫做 bucket。每个 bucket 有两部分:一个是键对象的引用,一个是值对象的引 用。由于,所有 bucket 结构和大小一致,我们可以通过偏移量来读取指定 bucket。
a = {}
a[“name”]=“gaoqi”
-
假设字典 a 对象创建完后,数组长度为 8:
-
我们要把”name”=”gaoqi”这个键值对放到字典对象 a 中,首先第一步需要计算 键”name”的散列值。Python 中可以通过 hash()来计算。
bin(hash(“name”))
‘-0b1010111101001110110101100100101’
- 由于数组长度为 8,我们可以拿计算出的散列值的最右边 3 位数字作为偏移量,即 “101”,十进制是数字 5。我们查看偏移量 5,对应的 bucket 是否为空。如果为空,则 将键值对放进去。如果不为空,则依次取右边 3 位作为偏移量,即“100”,十进制是数字4。再查看偏移量为 4 的 bucket 是否为空。直到找到为空的 bucket 将键值对放进去。流程图如下:
- 扩容:python 会根据散列表的拥挤程度扩容。“扩容”指的是:创造更大的数组,将原有内容 拷贝到新数组中。 接近 2/3 时,数组就会扩容。
7. 根据键查找‘键值对’的底层过程
- 我们明白了,一个键值对是如何存储到数组中的,根据键对象取到值对象,理解起来就 简单了。
a.get(“name”)
‘gaoqi’
- 当我们调用 a.get(“name”),就是根据键“name”查找到“键值对”,从而找到值 对象“gaoqi”。
- 第一步,我们仍然要计算“name”对象的散列值:
bin(hash(“name”))
‘-0b1010111101001110110101100100101’
和存储的底层流程算法一致,也是依次取散列值的不同位置的数字。 假设数组长度为 8,我们可以拿计算出的散列值的最右边 3 位数字作为偏移量,即“101”,十进制是数字 5。我们查看偏移量 5,对应的 bucket 是否为空。如果为空,则返回 None。如果不为空, 则将这个 bucket 的键对象计算对应散列值,和我们的散列值进行比较,如果相等。则将对 应“值对象”返回。如果不相等,则再依次取其他几位数字,重新计算偏移量。依次取完后, 仍然没有找到。则返回 None。流程图如下:
用法总结:
- 键必须可散列
(1) 数字、字符串、元组,都是可散列的。
(2) 自定义对象需要支持下面三点:
支持 hash()函数
支持通过__eq__()方法检测相等性。
若 a==b 为真,则 hash(a)==hash(b)也为真。 - 字典在内存中开销巨大,典型的空间换时间。
- 键查询速度很快
- 往字典里面添加新建可能导致扩容,导致散列表中键的次序变化。因此,不要在遍历字 典的同时进行字典的修改。
二、集合
- 集合是无序可变的,元素不能重复。实际上,集合底层是字典实现,集合的所有元素都是字典中的‘键对象’,因此是不能重复的且唯一的。
1. 集合的创建和删除
- 使用{}创建集合对象,并使用add()方法添加元素
a={3,5,7}
print(a)
a.add(9)
print(a)
运行结果:
{3, 5, 7}
{9, 3, 5, 7}
- 使用set(),将列表、元组等可迭代对象转成集合,如果原来数据存在数据重复,则保留一个
a=['a','b','c','b'] 这是个列表
b=set(a)
print(b)
运行结果:
['b','a','c']
- remove()删除指定元素;clear清空整个集合
a.remove(20)
print(a)
a.clear()
运行结果:
{10,30,40,50}
{}
2. 集合相关操作
- python对集合也提供了交集、并集、差集等运算。
- a | b ==a.uniom(b). 并集
- a & b ==a.intersection(b) 交集
- a - b ==a.difference(b) 差集
三、选择表达式
1. 单分支选择结构
-
if 条件表达式:(逻辑表达式,关系表达式,算数表达式均可)
语句/语句快(可以一条,可以多条)
pass -
在选择和循环结构中,条件表达式的值为False的情况如下:
False、0、0.0、空值None、空序列对象、空range对象、空迭代对象。
其他情况均为Ture。(字符串"False"为Ture)
注意:条件表达式中,不能有赋值操作符’ = ’
2. 双分支结构
if 条件表达式:
语句1/语句块1
else
语句2/语句块2
3. 三元条件运算符
- (条件为真的值) if (条件表达式) else (条件为假的值)
4. 多分支结构
if 条件表达式
语句/语句块
elif 条件表达式
语句/语句块
…
elif 条件表达式
语句/语句块
else
5. 选择结构的嵌套
x=int(input('请输入横坐标'))
y=int(input('请输入纵坐标'))
if x>0:
if y>0:
print('坐标({0},{1})在第一象限'.format(x,y))
elif y<0:
print('坐标({0},{1})在第四象限'.format(x, y))
else:
print('坐标({0},{1})在坐标轴上'.format(x, y))
elif x<0:
if y>0:
print('坐标({0},{1})在第二象限'.format(x,y))
elif y<0:
print('坐标({0},{1})在第三象限'.format(x, y))
else:
print('坐标({0},{1})在坐标轴上'.format(x, y))
else:
print('坐标({0},{1})在坐标轴上'.format(x, y))
#输入成绩0-100,90以上是A,80-89是B,70-79是C,60-69是D,60以下是E
score=int(input('输入分数'))
degree='ABCDE'
num=0
if score>100 or score<0:
print('请输入一个0-100的数字')
score = int(input('输入分数'))
pass
else:
num=score//10 # //向负方向取整
if num<6:
num=5
elif num>9:
num=9
print('分数是{0},等级是{1}'.format(score,degree[9-num]))