艰难学习Python Ex.41困惑关于For循环

时间:2013-08-31 02:54:02

标签: python-2.7

我无法理解其中一个for循环如何在Learn Python the Hard Way ex.41中运行。 http://learnpythonthehardway.org/book/ex41.html以下是课程中的代码。

我感到困惑的循环是for i in range(0, snippet.count("@@@")): 它是否在0到snippet的范围内迭代(其中有6个片段),并添加“@@@”计数的额外值?那么对于下一行代码param_count = random.randint(1,3),是否应用了“@@@”的额外值?或者我离开了!?

干杯 达伦

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 '***'."
}

# do they want to drill phrases first
PHRASE_FIRST = False
if len(sys.argv) == 2 and sys.argv[1] == "english":
    PHRASE_FIRST = True

# load up the words from the website
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[:]

        # fake class names
        for word in class_names:
            result = result.replace("%%%", word, 1)

        # fake other names
        for word in other_names:
            result = result.replace("***", word, 1)

        # fake parameter lists
        for word in param_names:
           result = result.replace("@@@", word, 1)

        results.append(result)

    return results


# keep going until they hit CTRL-D
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"

2 个答案:

答案 0 :(得分:3)

snippet.count("@@@")返回"@@@" snippet中显示的次数"@@@"

如果{{1}}出现6次,则for循环从0迭代到6。

答案 1 :(得分:0)

“try except”块运行程序直到用户点击^ D。

“While True”循环“try”将PHRASES dictonary中的键列表存储到片段中。每次键的顺序不同(因为随机方法)。 “for loop”内部的“while循环”是遍历每个片段,并在该片段的键和值上调用convert方法。

所有“转换方法”用来替换该键和值的%%%,***和@@@,并使用url列表中的随机词返回一个列表(results)由两个字符串组成:一个由钥匙制成,一个由价值制成。

然后程序打印其中一个字符串作为问题,然后获取用户输入(使用raw_input(“>”)),但无论用户输入什么,它都会打印另一个返回的字符串作为答案。

在内部转换方法中,我们有三个不同的列表:class_names,other_names和param_names。 要创建class_names,程序会计算该键的%%% isnide数(或值,但它们中的%%%数相同)。 class_names将是%%%计数大小的随机单词列表。

other_names是一个随机的单词列表。多少字?在密钥中找到***的数量(或者值,无关紧要,因为它们在任何一对中都是相同的)

param_names是一个字符串列表,其大小为@@@ found。每个字符串由一个,两个或三个不同的单词组成,分别为。

'result'是一个字符串。该程序遍历三个列表(class_names,param_names和other_names),并将结果字符串中的内容替换为已为其准备的内容。然后将其附加到结果列表中。 (对于句子中的句子,短语:)循环运行两次,因为'snippet'和'短语'是两个不同的字符串。因此,'结果'字符串正在进行两次(一个用于回答问题1)。

我把这个程序的一部分放到一个较小的子程序中,以阐明如何创建url中随机单词的特定大小的列表:

https://github.com/MahshidZ/python-recepies/blob/master/random_word_set.py

最后,我建议将print语句放在代码中需要更好理解的位置。一个例子,对于这段代码,我打印了一些变量来准确地得到正在发生的事情。这是一个没有调试器的调试的好方法:(在我的代码中查找布尔变量DEBUG)

DEBUG = 1
if DEBUG:
  print "snippet: " , snippet
  print "phrase: ", phrase
  print "class names: ", class_names
  print "other names: " , other_names
  print "param names: ", param_names