关于 python 中的字典的一点疑问,请各位大神帮忙解答

2016-09-14 10:51:13 +08:00
 xuweitiger

小弟是菜鸡一枚,最近在学习 python ,按着《 Head_First_Python 》中的样例代码一点一点的写,在写到 Chapter 7 的时候遇到一点疑惑,烦请各位大神解答.

问题描述: 这是一个记录运动员姓名,出生日期,以及每次训练成绩的问题, 需要处理的文本文件的格式是酱紫的:

Sarah Sweeney,2002-6-17,2:58,2.58,2:39,2-25,2-55,2:54,2.18,2:55,2:55,2:22,2-21,2.22

这是其中的一个文件,里面记录的运动员的姓名,出生日期,以及每次训练的成绩(时间),类似的文件总共有四个。

我的问题是,在最后输出阶段, data 是经过程序处理返回的一个字典,里面的 key 是姓名, value 是运动成绩,但是为何可以通过

data[each_athlete].dob

这种方式调用到出生日期呢? 按理说 data 不应该只包含运动员的姓名和成绩吗?

以下是代码:

先是定义了一个类:

class AthleteList(list):
    def __init__(self,a_name,a_dob=None,a_times=[]):
        list.__init__([])
        self.name = a_name
        self.dob = a_dob
        self.extend(a_times)

然后引用这个类:

import pickle
from athletelist import AthleteList
def get_coach_data(filename):
    try:
        with open(filename) as f:
            data = f.readline()
            templ = data.strip().split(',')
            return(AthleteList(templ.pop(0),templ.pop(0),templ))
    except IOError as ioerr:
        print('File error: ' + str(ioerr))
        return(None)
        
def put_to_store(files_list):
    all_athletes = {}
    for each_file in files_list:
        ath = get_coach_data(each_file)
        all_athletes[ath.name] = ath
    try:
        with open('athletes.pickle','wb') as athf:
            pickle.dump(all_athletes,athf)
    except IOError as ioerr:
        print('File error (put_and_store):' + str(ioerr))
    return(all_athletes)

def get_from_store():
    all_athletes = {}
    try:
        with open('athletes.pickle','rb') as athf:
            all_athletes = pickle.load(athf)
    except IOError as ioerr:
        print('File error(get_from_store):' + str(ioerr))
    return(all_athletes)

然后是输出:

print('-----------------第一段输出------------------------')

the_files = ['sarah2.txt','james2.txt','mikey2.txt','julie2.txt']
data = put_to_store(the_files)
print(data)

第一段输出的结果:

{'Julie Jones': ['2.59', '2.11', '2:11', '2:23', '3-10', '2-23', '3:10', '3.21', '3-21', '3.01', '3.02', '2:59'], 'James Lee': ['2-34', '3:21', '2.34', '2.45', '3.01', '2:01', '2:01', '3:10', '2-22', '2-01', '2.01', '2:16'], 'Sarah Sweeney': ['2:58', '2.58', '2:39', '2-25', '2-55', '2:54', '2.18', '2:55', '2:55', '2:22', '2-21', '2.22'], 'Mikey McManus': ['2:22', '3.01', '3:01', '3.02', '3:02', '3.02', '3:22', '2.49', '2:38', '2:40', '2.22', '2-31']}
print('-----------------第二段输出------------------------')

for each_athlete in data:
    print(data[each_athlete].name + ' ' + data[each_athlete].dob)

第二段输出结果:

Julie Jones 2002-8-17
James Lee 2002-3-14
Sarah Sweeney 2002-6-17
Mikey McManus 2002-2-24

print('------------------第三段输出-------------------------')

data_copy = get_from_store()
for each_athlete in data_copy:
    print(data_copy[each_athlete].name + ' ' + data_copy[each_athlete].dob)

第三段输出的结果:

Julie Jones 2002-8-17
Sarah Sweeney 2002-6-17
James Lee 2002-3-14
Mikey McManus 2002-2-24

2266 次点击
所在节点    Python
8 条回复
xuweitiger
2016-09-14 11:09:40 +08:00
请各位大哥大姐帮忙解答,多谢
aiver
2016-09-14 11:13:52 +08:00
首先, AthleteList 类里面定义了三个字段: name , dob 和成绩 list , dob 字段就是出生日期。然后在 get_coach_data 函数进行 data 填充时把“ Sarah Sweeney,2002-6-17,2:58,2.58,2:39,2-25,2-55,2:54,2.18,2:55,2:55,2:22,2-21,2.22 ”格式数据的第二个字段即“ 2002-6-17 ”填充到了 AthleteList 的 dob 字段,返回的是一个以 name 为键, AthleteList 实例为值的字典,所以在输出时可以通过 name 键找到相应的 AthleteList 实例,实例里面自然可以通过 dob 字段来访问到出生日期了。
xuweitiger
2016-09-14 11:23:37 +08:00
@aiver 那为何在第一段输出时,直接 print ( data )的时候,输出结果中没有 dob 呢?
lcc4376
2016-09-14 11:24:48 +08:00
def get_coach_data(filename):
裡 templ = data.strip().split(',')
return(AthleteList(templ.pop(0),templ.pop(0),templ)) <-----第二個 templ.pop(0)就是 dob 了
lcc4376
2016-09-14 11:26:20 +08:00
@xuweitiger 第一段並沒有用 get_coach_data 去 split(',')
lxy
2016-09-14 11:50:54 +08:00
AthleteList 继承了 list ,做了一点改造。 data 是个字典,字典 value 是 AthleteList 对象。 print 的时候好像默认调用对象的__str__ 函数,而 AthleteList 并没有改写该函数,所以调用的是它的父类 list 的 __str__,打印的即 self.extend(a_times) 中加入的内容。
aiver
2016-09-14 12:32:09 +08:00
@xuweitiger 这个其实你要看 AthleteList 的定义,它只是继承了 list 但并未重写 list 中的方法, print 输出默认是调用输出类型的__str__内置函数,因为 AthleteList 并未重新__str__函数,所以调用的父类 list 的__str__函数, list 的__str__输出就是它自身的内容,对于 AthleteList 就是输出 self.extend()中的内容。如果按照完整的输出的话,其实第一部分应该输出内容如下:{'Julie Jones':{'name':'Julie Jones', 'dob':'2002-6-17', ['2.59', '2.11', '2:11', '2:23', '3-10', '2-23', '3:10', '3.21', '3-21', '3.01', '3.02', '2:59'],}, {...}}。
xuweitiger
2016-09-14 14:46:10 +08:00
@aiver
@lcc4376
@lxy
感谢三位大神回复

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/306129

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX