learn python the hard way结尾_Learn Python The Hard Way 习题41详解

Learn Python The Hard Way 习题41详解

标签: Python 博客

博主最近在学习Python,看的书是Learn Python The Hard Way(Third Edition), 前40道习题没有什么难度,但是看到习题41的时候,由于出现了很多新函数和新名字以及印刷错误,竟然没看懂这道题的目的。查询了一些函数的用法之后,现在把这道题搞清楚了,分享出来,希望能对其他新手有所帮助。

1、印刷错误

译者:王巍巍

版次:2014年11月第1版

印刷时间:2015年5月北京第3次印刷

具体错误:#fake class namesfor word in class_names: result = result.replace('%%%', word, 1)# fake other namesfor word in other_names: result = result.replace('***', word, 1)# fake parameter listsfor word in param_names: result = result.replace('@@@', word, 1)results.append(result)1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

上面的一段代码在书中是Line47~Line59,这一段代码应该全部再缩进一格,即这段代码处于for sentence in snippet, phrase:的循环中,具体可以参考Exercise 41: Learning To Speak Object Oriented

2、本节新函数介绍

2.1strip()

首次出现位置:Line30

函数原型:str.strip([chars])

参数:chars – 移除字符串头尾指定的字符,默认为空格

返回值:返回移除字符串头尾指定的字符生成的新字符串,移除的是chars中的任意字符

实例:

>>>a = ' 123'>>>print a.strip()123>>>a ='123abc'>>>print a.strip('1cb')23a1

2

3

4

5

6

2.2capitalize()

首次出现位置:Line34

函数原型:str.capitalize()

参数:None

返回值:返回一个首字母大写,其余字母小写的字符串

实例:

>>>a = 'this is a book called HARRY POTTER'>>>print a.capitalize()This is a book called harry potter1

2

3

2.3random.sample()

首次出现位置:Line35

函数原型:random.sample(sequence, k)

参数:sequence是一个list,k是一个整数

返回值:返回一个list,该list由sequence中随机的k个元素组成,sequence不变

实例:

>>>list = [1,2,3,4,5,6,7,8,9,10]>>>slice = random.sample(list,5)>>>print slice[3,6,8,2,4]#截取的序列元素并没有顺序1

2

3

4

2.4count()

首次出现位置:Line35

函数原型:str.count(sub, start= 0,end=len(str))

参数:sub – 搜索的子字符串

start – 字符串开始搜索的位置,默认为第一个字符,第一个字符索引值为0

end – 字符串中结束搜索的位置,字符中第一个字符的索引为0,默认为字符串的最后一个位置

返回值:返回子字符串在字符串中出现的次数

实例:

>>>a = 'this is a book called HARRY POTTER'>>>print a.count('i')21

2

3

2.5join()

首次出现位置:Line42

函数原型:str.join(sequence)

参数:sequence – 要连接元素的list

返回值:返回通过str连接序列中元素后生成的字符串

实例:

>>>str = '-'>>>seq = ['a','b','c']>>>print str.join(seq)a-b-c1

2

3

4

2.6replace()

首次出现位置:Line49

函数原型:str.replace(old, new[, max])

参数:old – 将被替换的子字符串。

new – 新字符串,用于替换old子字符串。

max – 可选字符串, 替换不超过 max 次

返回值:返回以new代替old不超过max次的新字符串

实例:

>>>a = 'this is a book called HARRY POTTER'>>>print a.replace('is', 'was', 1)thwas is a book called HARRY POTTER>>>print a.replace('is', 'was')thwas was a book called HARRY POTTER1

2

3

4

5

2.7keys()

首次出现位置:Line67

函数原型:dict.keys()

参数:None

返回值:返回一个字典所有的键

实例:

>>>dict = {'name':'moverzp', 'age':'23', 'height':'180'}>>>print dict.keys()['age','name','height']1

2

3

2.8shuffle()

首次出现位置:Line68

函数原型:random.shuffle(lst)

参数:lst – 可以是一个序列或者元组

返回值:None. 直接修改lst的顺序

实例:

>>>list = [1, 2, 3, 4]>>>random.shuffle(list)>>>print list[3, 1, 4, 2]>>>random.shuffle(list)>>>print list[1, 4, 2, 3]1

2

3

4

5

6

7

3、代码详解

这段脚本的作用其实就是在设定网页中随机取出单词当做类的名字,函数的名字或者参数的名字。具体操作就是把PHRASES中的”%%%”,”***”,”@@@”替换成随机选择的单词而已,PHRASE_FIRST决定先打印单词替换符号后key-value(键值对)中的key还是value。

分别以流程图和代码注释说明。流程图如下:

Created with Rapha?l 2.1.0开始从网页读取单词并写入WORDS提取PHRASE中的键存入snippets并打乱顺序从snippets中取出一个元素snippet取出PHRASE中snippet键对应的值phrase(进入convert函数)phrase中有几个“%%%”就随机从WORDS中取几个单词放入class_names将class_names中的单词首字母大写phrase中有几个“***”就随机从WORDS中取几个单词放入other_names将other_names中的单词首字母大写phrase中是否有“@@@”在[1,3]取随机数param_count在WORDS中取param_count个单词,将其用', '连接起来并写入param_namesfor sentence in snippet, phraseresult浅拷贝sentence用class_names中的单词替换result中的“%%%”用other_names中的单词替换result中的“***”用param_names中的单词替换result中的“@@@”把result作为一个元素加入results中snippet和phrase处理完以后,返回results(退出convert函数)根据PHRASE_FIRST的值决定处理后snippet和phrase的打印顺序,即先打印字典中的键还是值结束yesno代码分析如下:

import randomfrom urllib import urlopenimport sysWORD_URL = 'http://learncodethehardway.org/words.txt'#该网页中全是单独成行的单词WORDS = []PHRASES = { #编写脚本时应该写的代码为key,其解释为value 'class %%%(%%%):': #%%%表示类名 'Make a class named %%% that is-a %%%.', 'class %%%(object):\n\tdef __init__(self, ***)' : 'class %%% has-a __init__ that takes self and *** parameters.', 'class %%%(object):\n\tdef ***(self, @@@)': 'class %%% has-a function named *** that takes self and @@@ parameters.', '*** = %%%()': 'Set *** to an instance of class %%%.', '***.***(@@@)': 'From *** get the *** function, and call it with parameters self, @@@.', '***.*** = '***'': 'From *** get the *** attribute and set it to '***'.'}# do they want to drill phrases firstPHRASE_FIRST = False #False表示先打印key,按下任意键后再打印valueif len(sys.argv) == 2 and sys.argv[1] == 'english': PHRASE_FIRST = True #True表示先打印value,按下任意键后再打印key# load up the words from the websitefor word in urlopen(WORD_URL).readlines(): #一行一行从网页中读取数据 WORDS.append(word.strip()) #删除每一行开始和结尾的空格,只留下单词并加入到words列表中#先从下面的try开始看,调用到该函数时再看回来def convert(snippet, phrase): #我们以传入的具体参数来分析 class_names = [w.capitalize() for w in #从WORDS序列中随机取出1个单词首字母大写后赋值给class_names,此处假设为'Actor' random.sample(WORDS, snippet.count('%%%'))] other_names = random.sample(WORDS, snippet.count('***')) #从WORDS序列中随机取出一个单词赋值给other_names,此处假设为'dinner' results = [] param_names = [] for i in range(0, snippet.count('@@@')): #只循环一次 param_count = random.randint(1,3) #假设随机到了2 param_names.append(', '.join(random.sample(WORDS, param_count))) #从WORDS中随机取出2个单词用逗号和空格连接起来,放在param_names中,假设为'cook, donkey' for sentence in snippet, phrase: #循环两次,只讲解第一次循环 result = sentence[:] #深拷贝,result = 'class %%%(object):\n\tdef ***(self, @@@)' # fake class names for word in class_names: result = result.replace('%%%', word, 1) #result = 'class Actor(object):\n\tdef ***(self, @@@)' # fake other names for word in other_names: result = result.replace('***', word, 1) #result = 'class Actor(object):\n\tdef dinner(self, @@@)' # fake parameter lists for word in param_names: result = result.replace('@@@', word, 1) #result = 'class Actor(object):\n\tdef dinner(self, cook, donkey)' results.append(result) #添加到results序列中 return results #results = ['class Actor(object):\n\tdef dinner(self, cook, donkey)', 'class Actor has-a function named dinner that takes self and cook, donkey parameters.']# keep going until they hit CTRL-Dtry: while True: snippets = PHRASES.keys() #提取PHRASES中的键 random.shuffle(snippets) #打乱snippets中元素(PHRASES中键)顺序 for snippet in snippets: #取一个键来操作,假设取到'class %%%(object):\n\tdef ***(self, @@@)' phrase = PHRASES[snippet] #取出值'class %%% has-a function named *** that takes self and @@@ parameters.' question, answer = convert(snippet, phrase) #现在去上面看这个函数的定义 if PHRASE_FIRST: #先打印value,再打印key question, answer = answer, question print question raw_input('> ') print 'ANSWER: %s\n\n' % answerexcept EOFError: print '\nBye'1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

4、总结当遇到看不懂的代码的时候,可以将其打印出来,用铅笔一行一行写注释去分析

遇到不懂的函数的时候一定要去网上搜索函数原型,至少要了解参数以及返回值

可以用print一步一步来调试

必要的时候画流程图

可以带入具体参数来分析

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值