在本章中,你将学习能够将相关信息关联起来的Python字典。你将学习如何访问和修改字典中的信息。鉴于字典可存储的信息量几乎不受限制,因此我们会演示如何遍
历字典中的数据。另外,你还将学习存储字典的列表、存储列表的字典和存储字典的字典。
理解字典后,你就能够更准确地为各种真实物体建模。你可以创建一个表示人的字典,然后想在其中存储多少信息就存储多少信息:姓名、年龄、地址、职业以及要
描述的任何方面。你还能够存储任意两种相关的信息,如一系列单词及其含义,一系列人名及其喜欢的数字,以及一系列山脉及其海拔等。
有时候,需要将一系列字典存储在列表中,或将列表作为值存储在字典中,这称为嵌套 。你可以在列表中嵌套字典、在字典中嵌套列表甚至在字典中嵌套字典。正如下面的示例
将演示的,嵌套是一项强大的功能。
字典列表
字典alien_0 包含一个外星人的各种信息,但无法存储第二个外星人的信息,更别说屏幕上全部外星人的信息了。如何管理成群结队的外星人呢?一种办法是创建一个外星人列
表,其中每个外星人都是一个字典,包含有关该外星人的各种信息。例如,下面的代码创建一个包含三个外星人的列表:
alien_0 = {'color': 'green', 'points': 5}
alien_1 = {'color': 'yellow', 'points': 10}
alien_2 = {'color': 'red', 'points': 15}
❶ aliens = [alien_0, alien_1, alien_2]
for alien in aliens:
print(alien)
我们首先创建了三个字典,其中每个字典都表示一个外星人。在❶处,我们将这些字典都放到一个名为aliens 的列表中。最后,我们遍历这个列表,并将每个外星人都打印出
来:
{'color': 'green', 'points': 5}
{'color': 'yellow', 'points': 10}
{'color': 'red', 'points': 15}
更符合现实的情形是,外星人不止三个,且每个外星人都是使用代码自动生成的。在下面的示例中,我们使用range() 生成了30个外星人:
# 创建一个用于存储外星人的空列表
aliens = []
# 创建30个绿色的外星人
❶ for alien_number in range(30):
❷ new_alien = {'color': 'green', 'points': 5, 'speed': 'slow'}
❸ aliens.append(new_alien)
# 显示前五个外星人
❹ for alien in aliens[:5]:
print(alien)
print("...")
# 显示创建了多少个外星人
❺ print("Total number of aliens: " + str(len(aliens)))
在这个示例中,首先创建了一个空列表,用于存储接下来将创建的所有外星人。在❶处,range() 返回一系列数字,其唯一的用途是告诉Python我们要重复这个循环多少次。每
次执行这个循环时,都创建一个外星人(见❷),并将其附加到列表aliens 末尾(见❸)。在❹处,使用一个切片来打印前五个外星人;在❺处,打印列表的长度,以证明确
实创建了30个外星人:
{'speed': 'slow', 'color': 'green', 'points': 5}
{'speed': 'slow', 'color': 'green', 'points': 5}
{'speed': 'slow', 'color': 'green', 'points': 5}
{'speed': 'slow', 'color': 'green', 'points': 5}
{'speed': 'slow', 'color': 'green', 'points': 5}
...
Total number of aliens: 30
这些外星人都具有相同的特征,但在Python看来,每个外星人都是独立的,这让我们能够独立地修改每个外星人。
在什么情况下需要处理成群结队的外星人呢?想象一下,可能随着游戏的进行,有些外星人会变色且移动速度会加快。必要时,我们可以使用for 循环和if 语句来修改某些外星
人的颜色。例如,要将前三个外星人修改为黄色的、速度为中等且值10个点,可以这样做:
# 创建一个用于存储外星人的空列表
aliens = []
# 创建30个绿色的外星人
for alien_number in range (0,30):
new_alien = {'color': 'green', 'points': 5, 'speed': 'slow'}
aliens.append(new_alien)
for alien in aliens[0:3]:
if alien['color'] == 'green':
alien['color'] = 'yellow'
alien['speed'] = 'medium'
alien['points'] = 10
# 显示前五个外星人
for alien in aliens[0:5]:
print(alien)
print("...")
鉴于我们要修改前三个外星人,需要遍历一个只包含这些外星人的切片。当前,所有外星人都是绿色的,但情况并非总是如此,因此我们编写了一条if 语句来确保只修改绿色外
星人。如果外星人是绿色的,我们就将其颜色改为’yellow’ ,将其速度改为’medium’ ,并将其点数改为10 ,如下面的输出所示:
{'speed': 'medium', 'color': 'yellow', 'points': 10}
{'speed': 'medium', 'color': 'yellow', 'points': 10}
{'speed': 'medium', 'color': 'yellow', 'points': 10}
{'speed': 'slow', 'color': 'green', 'points': 5}
{'speed': 'slow', 'color': 'green', 'points': 5}
...
你可以进一步扩展这个循环,在其中添加一个elif 代码块,将黄色外星人改为移动速度快且值15个点的红色外星人,如下所示(这里只列出了循环,而没有列出整个程序):
for alien in aliens[0:3]:
if alien['color'] == 'green':
alien['color'] = 'yellow'
alien['speed'] = 'medium'
alien['points'] = 10
elif alien['color'] == 'yellow':
alien['color'] = 'red'
alien['speed'] = 'fast'
alien['points'] = 15
经常需要在列表中包含大量的字典,而其中每个字典都包含特定对象的众多信息。例如,你可能需要为网站的每个用户创建一个字典(就像6.3.1节的user.py中那样),并将这些
字典存储在一个名为users 的列表中。在这个列表中,所有字典的结构都相同,因此你可以遍历这个列表,并以相同的方式处理其中的每个字典。
在字典中存储列表
有时候,需要将列表存储在字典中,而不是将字典存储在列表中。例如,你如何描述顾客点的比萨呢?如果使用列表,只能存储要添加的比萨配料;但如果使用字典,就不仅可
在其中包含配料列表,还可包含其他有关比萨的描述。
在下面的示例中,存储了比萨的两方面信息:外皮类型和配料列表。其中的配料列表是一个与键’toppings’ 相关联的值。要访问该列表,我们使用字典名和键’toppings’
,就像访问字典中的其他值一样。这将返回一个配料列表,而不是单个值:
# 存储所点比萨的信息
❶ pizza = {
'crust': 'thick',
'toppings': ['mushrooms', 'extr a cheese'],
}
# 概述所点的比萨
❷ print("You ordered a " + pizza['cru st'] + "-crust pizza " +
"with the following toppings:")
❸ for topping in pizza['toppings']:
print("\t" + topping)
我们首先创建了一个字典,其中存储了有关顾客所点比萨的信息(见❶)。在这个字典中,一个键是’crust’ ,与之相关联的值是字符串’thick’ ;下一个键是’toppings’
,与之相关联的值是一个列表,其中存储了顾客要求添加的所有配料。制作前我们概述了顾客所点的比萨(见❷)。为打印配料,我们编写了一个for 循环(见❸)。为访问配
料列表,我们使用了键’toppings’ ,这样Python将从字典中提取配料列表。
下面的输出概述了要制作的比萨:
You ordered a thick-crust pizza with the following toppings:
mushrooms
extra cheese
每当需要在字典中将一个键关联到多个值时,都可以在字典中嵌套一个列表。在本章前面有关喜欢的编程语言的示例中,如果将每个人的回答都存储在一个列表中,被调查者就
可选择多种喜欢的语言。在这种情况下,当我们遍历字典时,与每个被调查者相关联的都是一个语言列表,而不是一种语言;因此,在遍历该字典的for 循环中,我们需要再使
用一个for 循环来遍历与被调查者相关联的语言列表:
❶ favorite_languages = {
'jen': ['python', 'ruby'],
'sarah': ['c'],
'edward': ['ruby', 'go'],
'phil': ['python', 'haskell'],
}
❷ for name, languages in favorite_languages.items():
print("\n" + name.title() + "'s favorite languages are:")
❸ for language in languages:
print("\t" + language.title())
正如你看到的,现在与每个名字相关联的值都是一个列表(见❶)。请注意,有些人喜欢的语言只有一种,而有些人有多种。遍历字典时(见❷),我们使用了变量languages
来依次存储字典中的每个值,因为我们知道这些值都是列表。在遍历字典的主循环中,我们又使用了一个for 循环来遍历每个人喜欢的语言列表(见❸)。现在,每个人想列出
多少种喜欢的语言都可以:
Jen's favorite languages are:
Python
Ruby
Sarah's favorite languages are:
C
Phil's favorite languages are:
Python
Haskell
Edward's favorite languages are:
Ruby
Go
为进一步改进这个程序,可在遍历字典的for 循环开头添加一条if 语句,通过查看len(languages) 的值来确定当前的被调查者喜欢的语言是否有多种。如果他喜欢的语言有
多种,就像以前一样显示输出;如果只有一种,就相应修改输出的措辞,如显示Sarah’s favorite language is C 。
注意 列表和字典的嵌套层级不应太多。如果嵌套层级比前面的示例多得多,很可能有更简单的解决问题的方案。
在字典中存储字典
可在字典中嵌套字典,但这样做时,代码可能很快复杂起来。例如,如果有多个网站用户,每个都有独特的用户名,可在字典中将用户名作为键,然后将每位用户的信息存储在
一个字典中,并将该字典作为与用户名相关联的值。在下面的程序中,对于每位用户,我们都存储了其三项信息:名、姓和居住地;为访问这些信息,我们遍历所有的用户名,
并访问与每个用户名相关联的信息字典:
users = {
'aeinstein': {
'first': 'albert',
'last': 'einstein',
'location': 'princeton',
},
'mcurie': {
'first': 'marie',
'last': 'curie',
'location': 'paris',
},
}
❶ for username, user_info in users.items():
❷ print("\nUsername: " + username)
❸ full_name = user_info['first'] + " " + user_info['last']
location = user_info['location']
❹ print("\tFull name: " + full_name.title())
print("\tLocation: " + location.title())
我们首先定义了一个名为users 的字典,其中包含两个键:用户名’aeinstein’ 和’mcurie’ ;与每个键相关联的值都是一个字典,其中包含用户的名、姓和居住地。在❶
处,我们遍历字典users ,让Python依次将每个键存储在变量username 中,并依次将与当前键相关联的字典存储在变量user_info 中。在主循环内部的❷处,我们将用户名
打印出来。
在❸处,我们开始访问内部的字典。变量user_info 包含用户信息字典,而该字典包含三个键:‘first’ 、‘last’ 和’location’ ;对于每位用户,我们都使用这些键来
生成整洁的姓名和居住地,然后打印有关用户的简要信息(见❹):
Username: aeinstein
Full name: Albert Einstein
Location: Princeton
Username: mcurie
Full name: Marie Curie
Location: Paris
请注意,表示每位用户的字典的结构都相同,虽然Python并没有这样的要求,但这使得嵌套的字典处理起来更容易。倘若表示每位用户的字典都包含不同的键,for 循环内部的
代码将更复杂。