我是Python的新手,实际上这是我的wk2试图按照这本书,艰难地学习Python 。
对于ex41.py中的代码,我有两个问题。
我不太明白为什么它有以下声明:
for i in range(0, snippet.count("@@@")):
因为snippet.count("@@@")
总是1,所以我只有1个值,为什么我们为什么要循环呢?为什么我们不能直接设置变量param_count
和param_names
?
我也很困惑:
for word in param_names:
result = result.replace("@@@", word, 1)
我知道param_names
是一个最多包含三个单词的单词列表,但由于@@@
只在代码段和短语中出现一次,它如何被替换超过1次?
谢谢,
import random
from urllib import urlopen
import sys
WORD_URL = "http://learncodethehardway.org/words.txt"
WORDS = []
PHRASES = {
"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 '***'."
}
PHRASE_FIRST = False
if len(sys.argv) == 2 and sys.argv[1] == "english":
PHRASE_FIRST = True
for word in urlopen(WORD_URL).readlines():
WORDS.append(word.strip())
def convert(snippet, phrase):
class_names = [w.capitalize() for w in
random.sample(WORDS, snippet.count("%%%"))]
other_names = random.sample(WORDS, snippet.count("***"))
results = []
param_names = []
for i in range(0, snippet.count("@@@")):
param_count = random.randint(1,3)
param_names.append(', '.join(random.sample(WORDS, param_count)))
for sentence in snippet, phrase:
result = sentence[:]
for word in class_names:
result = result.replace("%%%", word, 1)
for word in other_names:
result = result.replace("***", word, 1)
for word in param_names:
result = result.replace("@@@", word, 1)
results.append(result)
return results
try:
while True:
snippets = PHRASES.keys()
random.shuffle(snippets)
for snippet in snippets:
phrase = PHRASES[snippet]
question, answer = convert(snippet, phrase)
if PHRASE_FIRST:
question, answer = answer, question
print question
raw_input("> ")
print "ANSWER: %s\n\n" % answer
except EOFError:
print "\nBye"
答案 0 :(得分:2)
参考您的问题2(result = result.replace("@@@", word, 1)
):
通常情况下,您不会多次更换它。
使用可选参数" 1",只有第一个" @@@"被取代。
例如:
str = "Hello @@@ hello @@@."
print str.replace("@@@", "world", 1)
print str.replace("@@@", "world", 2)
Hello world hello @@@.
Hello world hello world.
但在这种情况下,它是一个循环,它会在每个循环中替换一个项目。
str = "Hello @@@ hello @@@."
Loop1
str = "Hello world hello @@@."
Loop2
str = "Hello world hello world."
最后,如果您想知道为什么@@@被一个单词替换。
看看Join String方法:
param_names.append(', '.join(random.sample(WORDS, param_count)))
这种方法基本上做的是将几个字符串转换为一个字符串。
print ', '.join(["Word1", "Word2"])
Word1, Word2
答案 1 :(得分:1)
param_names
包含后续参数名称,因此,如果您有['param1', 'param2']
字符串'(@@@, @@@)'
,您希望第一个替换为result.replace("@@@", 'param1', 1)
,那么它只会替换第一个@@@
的出现导致'(param1, @@@)'
。然后,您想要将第二个@@@
替换为'(param1, param2)'
。如果没有约束,您最终会得到'(param1, param1)'
,即所有出现的@@@
都将替换为第一个参数的名称。这同样适用于所有其他占位符(类和#34;其他")。
对于某些特定输入数据而言似乎是多余的事实并不意味着它通常是多余的。除了脚本中的示例之外,还需要其他可能的输入,否则算法将不正确。