零基础入门学习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()

posted @ 2018-04-13 16:33  DB-Engineer  阅读(257)  评论(0编辑  收藏  举报