零基础入门学习Python(26)--字典:当索引不好用时2
知识点
删除字典元素
能删单一的元素也能清空字典,清空只需一项操作。
显示删除一个字典用del命令,如下:
>>> dict1 = {'a':1,'b':2,'c':3}
>>> del dict1['a']
>>> dict1
{'b': 2, 'c': 3}
>>> del dict1
>>> dict1
Traceback (most recent call last):
File "<pyshell#5>", line 1, in <module>
dict1
NameError: name 'dict1' is not defined
字典key的特性
字典value可以是任何的python对象,既可以是标准的对象,也可以是用户定义的,但key不行。
两个重要的店需要记住:
1) 不允许同一个key出现两次。创建时如果同一个key被赋值两次,后一个value会被记住,如下:
>>> dict1 = {'a':1,'b':2,'c':3,'a':4}
>>> dict1['a']
4
2) key必须不可变,所以可以用数字,字符串或元组充当,而用列表就不行,如下:
>>> dict2 = {['a']:1,'b':2}
Traceback (most recent call last):
File "<pyshell#11>", line 1, in <module>
dict2 = {['a']:1,'b':2}
TypeError: unhashable type: 'list'
#改成元祖形式
>>> dict2 = {('a','c'):1,'b':2}
>>> dict2
{('a', 'c'): 1, 'b': 2}
>>> dict2['a','c']
1
字典的内置方法:
clear()
:清除字典内元素
示例:
>>> dict1
{'a': 4, 'b': 2, 'c': 3}
>>> dict1.clear()
>>> dict1
{}
copy()
:返回一个字典的浅拷贝
示例:
>>> a = {1:'one',2:'two',3:'three'}
>>> b = a.copy()
>>> b
{1: 'one', 2: 'two', 3: 'three'}
直接赋值和copy的区别
>>> dict1 = {'a':1,'b':2,'c':[1,2,3]}
>>> dict2 = dict1 #浅拷贝: 引用对象
>>> dict3 = dict1.copy() #浅拷贝:深拷贝父对象(一级目录),子对象(二级目录)不拷贝,还是引用
#修改数据:
>>> dict1['a'] = '一'
>>> dict1['c'].remove(1)
#输出结果对比:
>>> dict1
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> dict2
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> dict3
{'a': 1, 'b': 2, 'c': [2, 3]}
示例中 dict2其实是dict1的引用(别名),所以输出结果都是一致的。
dict3父对象进行了深拷贝,不会随dict1修改而修改,子对象是浅拷贝所以随dict1的修改而修改
fromkeys()
创建一个新字典,以序列seq中元素做字典的键,val为字典所有键对应的初始值
示例:
#语法
dict.fromkeys(seq[, value])
#参数
seq -- 字典键值列表。
value -- 可选参数, 设置键序列(seq)的值,默认为None
#示例:
>>> dict1 = {}
>>> dict1.fromkeys((1,2,3))
{1: None, 2: None, 3: None}
>>> dict1.fromkeys((1,2,3),'Number')
{1: 'Number', 2: 'Number', 3: 'Number'}
>>> dict1.fromkeys((1,2,3),('One','Two','Three'))
{1: ('One', 'Two', 'Three'), 2: ('One', 'Two', 'Three'), 3: ('One', 'Two', 'Three')}
>>> dict1.fromkeys((1,3),'数字')
{1: '数字', 3: '数字'}
get(key, default=None)
返回指定键的值,如果key不存在,可以返回None,或者指定的value
示例:
>>> dict1
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> dict1.get('a')
'一'
>>> print(dict1.get('d'))
None
>>> print(dict1.get('d','四'))
四
key in dict
如果键在字典dict里返回true,否则返回false
示例:
>>> dict1
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> 'a' in dict1
True
>>> 'd' in dict1
False
keys()
以列表返回一个字典所有的键
示例
>>> dict1
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> dict1.keys()
dict_keys(['a', 'b', 'c'])
values()
以列表返回字典中的所有值
示例:
>>> dict1.values()
dict_values(['一', 2, [2, 3]])
items()
以列表返回可遍历的(键, 值) 元组数组
示例:
>>> dict1.items()
dict_items([('a', '一'), ('b', 2), ('c', [2, 3])])
setdefault(key, default=None)
和get()类似, 但如果键不存在于字典中,将会添加键并将值设为default
示例:
>>> a
{1: 'one', 2: 'two', 3: 'three'}
>>> a.setdefault(1)
'one'
>>> a.setdefault('小白')
>>> a
{1: 'one', 2: 'two', 3: 'three', '小白': None}
>>> a.setdefault(5,'five')
'five'
>>> a
{1: 'one', 2: 'two', 3: 'three', '小白': None, 5: 'five'}
update(dict2)
把字典dict2的键/值对更新到dict1里
示例:
#参数
dict2 -- 添加到指定字典dict1里的字典。
>>> dict1
{'a': '一', 'b': 2, 'c': [2, 3]}
>>> dict2 ={'c':[1,2,3,4]}
>>> dict1.update(dict2)
>>> dict1
{'a': '一', 'b': 2, 'c': [1, 2, 3, 4]}
>>> dict3 = {'d':5678}
>>> dict1.update(dict3)
>>> dict1
{'a': '一', 'b': 2, 'c': [1, 2, 3, 4], 'd': 5678}
pop(key[,default])
删除字典给定键 key 所对应的值,返回值为被删除的值。key值必须给出。 否则,返回default值。
示例:
>>> a = {1: 'one', 2: 'two', 3: 'three', 4: 'Four'}
>>> a.pop(2)
'two'
>>> a
{1: 'one', 3: 'three', 4: 'Four'}
>>> a.pop()
Traceback (most recent call last):
File "<pyshell#44>", line 1, in <module>
a.pop()
TypeError: pop expected at least 1 arguments, got 0
>>> a.pop(2)
Traceback (most recent call last):
File "<pyshell#80>", line 1, in <module>
a.pop(2)
KeyError: 2
>>> a.pop(5,'f')
'f'
>>> a
{1: 'one', 3: 'three', 4: 'Four'}
>>> a.pop(4,'f')
'Four'
>>> a
{1: 'one', 3: 'three'}
popitem()
随机返回并删除字典中的一对键和值(一般删除末尾对)。
示例:
>>> a
{1: 'one', 3: 'three', 4: 'Four'}
>>> a.popitem()
(4, 'Four')
>>> a
{1: 'one', 3: 'three'}
>>> a = {}
>>> a.popitem()
Traceback (most recent call last):
File "<pyshell#85>", line 1, in <module>
a.popitem()
KeyError: 'popitem(): dictionary is empty'
课后习题
测试题
- Python的字典是否支持一键(key)多值(value)?
不支持,对相同的键再次赋值会将上一次的值直接覆盖。
>>> dict1 = {1:'one', 1:'yi'}
>>> dict1[1]
'yi'
- 在字典中,如果试图为一个不存在的键(key)赋值会怎么样?
会自动创建对应的键(key)并添加相应的值(value)进去
成员资格操作符(in和not in)可以检查一个元素是否存在序列中,当然也可以用来检查一个键(key)是否存在字典中,那么请问哪种的检查效率更高些?为什么
在字典中检查键(key)
是否存在
比在序列中检查是否存在
更高效。
因为字典的原理是使用哈希算法存储,一步到位,不需要使用查找算法进行匹配,因此时间复杂度是0(1),效率非常高。Python对键(key)和值(value)有没有类型限制?
Python对键(key)的要求相对严格一些,要求它们必须是可哈希的对象,不能是可变类型(包括变量、列表、字典本身等。)
但是Python对值是没有任何限制的,它们可以是任意的Python对象。请目测下边代码执行后,字典dict1的内容是什么?
>>> dict1.fromkeys((1, 2, 3), ('one', 'two', 'three'))
>>> dict1.fromkeys((1, 3), '数字')
执行完成后,字典dict1的内容是:{1:'数字',2:'数字'}
【注意】:fromkeys方法是直接创建一个新的字典,不要试图使用它来修改一个原有的字典,因为它会直接无情的把整个字典给覆盖掉。
- 如果你需要将字典
dict1 = {1:'one',2:'two',3:'three'}
拷贝到dict2,你应该这么做?
可以利用copy()方法
dict2 = dict1.copy()
用赋值的方法(dict2 = dict1),这样子做在python中只是将对象的引用拷贝过去而已。
看以下区别:
>>> a = {1:'one', 2:'two', 3:'three'}
>>> b = a.copy()
>>> c = a
>>> c[4] = 'four'
>>> c
{1: 'one', 2: 'two', 3: 'three', 4: 'four'}
>>> a
{1: 'one', 2: 'two', 3: 'three', 4: 'four'}
>>> b
{1: 'one', 2: 'two', 3: 'three'}
动动手
- 尝试编写一个用户登录程序(这次尝试将功能封装成函数),程序实现如图:
user_data = {}
def new_user():
prompt = '请输入用户名:'
while True:
name = input(prompt)
if name in user_data:
prompt = '此用户名已经被使用,请重新输入:'
continue
else:
break
passwd = input('请输入密码:')
user_data[name] = passwd
print('注册成功,赶紧试试登录吧^_^')
def old_user():
prompt = '请输入用户名:'
while True:
name = input(prompt)
if name not in user_data:
prompt = '您输入的用户名不存在,请重新输入:'
continue
else:
break
passwd = input('请输入密码:')
pwd = user_data.get(name)
if passwd == pwd:
print('欢迎进入XXOO系统,请点右上角的X结束程序!')
else:
print('密码错误!')
def showmenu():
prompt = '''
|--- 新建用户:N/n ---|
|--- 登录账号:E/e ---|
|--- 推出程序:Q/q ---|
|--- 请输入指令代码:'''
while True:
chosen = False
while not chosen:
choice = input(prompt)
if choice not in 'NnEeQq':
print('您输入的指令代码错误,请重新输入:')
else:
chosen = True
if choice == 'q' or choice == 'Q':
break
if choice == 'n' or choice == 'N':
new_user()
if choice == 'e' or choice == 'E':
old_user()
showmenu()