python_DAY04

一、字典

  • 字典是‘键值对’的无序可变序列,字典中的每个元素都是一个“键值对”,包含“键对象”和“值对象”。
    一个典型的字典定义方式:
    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. 键必须可散列
    (1) 数字、字符串、元组,都是可散列的。
    (2) 自定义对象需要支持下面三点:
    支持 hash()函数
    支持通过__eq__()方法检测相等性。
    若 a==b 为真,则 hash(a)==hash(b)也为真。
  2. 字典在内存中开销巨大,典型的空间换时间。
  3. 键查询速度很快
  4. 往字典里面添加新建可能导致扩容,导致散列表中键的次序变化。因此,不要在遍历字 典的同时进行字典的修改。

二、集合

  • 集合是无序可变的,元素不能重复。实际上,集合底层是字典实现,集合的所有元素都是字典中的‘键对象’,因此是不能重复的且唯一的。

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]))
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值