defloadWords(): """ Returns a list of valid words. Words are strings of lowercase letters. Depending on the size of the word list, this function may take a while to finish. """ print("Loading word list from file...") # inFile: file inFile = open(WORDLIST_FILENAME, 'r') # wordList: list of strings wordList = [] for line in inFile: wordList.append(line.strip().lower()) print(" ", len(wordList), "words loaded.") return wordList
defgetFrequencyDict(sequence): """ Returns a dictionary where the keys are elements of the sequence and the values are integer counts, for the number of times that an element is repeated in the sequence. sequence: string or list return: dictionary """ # freqs: dictionary (element_type -> int) freq = {} for x in sequence: freq[x] = freq.get(x,0) + 1 return freq
defgetWordScore(word, n): """ Returns the score for a word. Assumes the word is a valid word. The score for a word is the sum of the points for letters in the word, multiplied by the length of the word, PLUS 50 points if all n letters are used on the first turn. Letters are scored as in Scrabble; A is worth 1, B is worth 3, C is worth 3, D is worth 2, E is worth 1, and so on (see SCRABBLE_LETTER_VALUES) word: string (lowercase letters) n: integer (HAND_SIZE; i.e., hand size required for additional points) returns: int >= 0 """ # score = 0 for elm in word: score += SCRABBLE_LETTER_VALUES[elm] score *= len(word) if len(word) == n: score += 50 return score
displayHand(hand),对当前拼字尝试在控制台上进行输出,便于游戏玩家查看
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
defdisplayHand(hand): """ Displays the letters currently in the hand. For example: >>> displayHand({'a':1, 'x':2, 'l':3, 'e':1}) Should print out something like: a x x l l l e The order of the letters is unimportant. hand: dictionary (string -> int) """ for letter in hand.keys(): for j in range(hand[letter]): print(letter,end=" ") # print all on the same line print() # print an empty line
defdealHand(n): """ Returns a random hand containing n lowercase letters. At least n/3 the letters in the hand should be VOWELS. Hands are represented as dictionaries. The keys are letters and the values are the number of times the particular letter is repeated in that hand. n: int >= 0 returns: dictionary (string -> int) """ hand={} numVowels = n // 3 for i in range(numVowels): x = VOWELS[random.randrange(0,len(VOWELS))] hand[x] = hand.get(x, 0) + 1 for i in range(numVowels, n): x = CONSONANTS[random.randrange(0,len(CONSONANTS))] hand[x] = hand.get(x, 0) + 1 return hand
defupdateHand(hand, word): """ Assumes that 'hand' has all the letters in word. In other words, this assumes that however many times a letter appears in 'word', 'hand' has at least as many of that letter in it. Updates the hand: uses up the letters in the given word and returns the new hand, without those letters in it. Has no side effects: does not modify hand. word: string hand: dictionary (string -> int) returns: dictionary (string -> int) """ # TO DO ... <-- Remove this comment when you code this function new = hand.copy() for letter in word: new[letter] -= 1 return new
defisValidWord(word, hand, wordList): """ Returns True if word is in the wordList and is entirely composed of letters in the hand. Otherwise, returns False. Does not mutate hand or wordList. word: string hand: dictionary (string -> int) wordList: list of lowercase strings """ # TO DO ... <-- Remove this comment when you code this function if word notin wordList: returnFalse for a in word: if hand.get(a,0) < getFrequencyDict(word)[a]: returnFalse returnTrue
这个函数也可以直接用一行写完。但似乎在可读性和效率上,也没有一行写完的必要?
1 2
# 2rd solution : one line # return word in wordList and all(hand.get(a, 0) >= b for a, b in getFrequencyDict(word).items())
calculateHandlen(hand)
计算拼字猜测中,总计包含的字母数量
1 2 3 4 5 6 7 8 9 10 11 12
defcalculateHandlen(hand): """ Returns the length (number of letters) in the current hand. hand: dictionary (string-> int) returns: integer """ # TO DO... <-- Remove this comment when you code this function sum = 0 for i in range(len(hand.keys())): sum += list(hand.values())[i] return sum
这个比较适合一行写完
1 2
# 2rd solution: # return sum(x for x in hand.values())
defplayHand(hand, wordList, n): """ Allows the user to play the given hand, as follows: * The hand is displayed. * The user may input a word or a single period (the string ".") to indicate they're done playing * Invalid words are rejected, and a message is displayed asking the user to choose another word until they enter a valid word or "." * When a valid word is entered, it uses up letters from the hand. * After every valid word: the score for that word is displayed, the remaining letters in the hand are displayed, and the user is asked to input another word. * The sum of the word scores is displayed when the hand finishes. * The hand finishes when there are no more unused letters or the user inputs a "." hand: dictionary (string -> int) wordList: list of lowercase strings n: integer (HAND_SIZE; i.e., hand size required for additional points) """ score = 0 # As long as there are still letters left in the hand: while calculateHandlen(hand) > 0: # Display the hand print("Current Hand:",end = ' ') displayHand(hand) # Ask user for input newin = input("Enter word, or a '.' to indicate that you are finished: ") # If the input is a single period: if newin == ".": # End the game (break out of the loop) break # Otherwise (the input is not a single period): else: # If the word is not valid: if isValidWord(newin, hand, wordList) == False: # Reject invalid word (print a message followed by a blank line) print("Invalid word, please try again.") # Otherwise (the word is valid): else: # Tell the user how many points the word earned, and the updated total score, in one line followed by a blank line current_score = getWordScore(newin, n) score += current_score print(newin + " earned " + str (current_score) + " points. Total: " + str(score)) # Update the hand hand = updateHand(hand, newin) # Game is over (user entered a '.' or ran out of letters), so tell user the total score if newin == '.': print('Goodbye! Total score:', score, 'points.') else: print('Run out of letters. Total score:', score, 'points.')
defplayGame(wordList): """ Allow the user to play an arbitrary number of hands. 1) Asks the user to input 'n' or 'r' or 'e'. * If the user inputs 'n', let the user play a new (random) hand. * If the user inputs 'r', let the user play the last hand again. * If the user inputs 'e', exit the game. * If the user inputs anything else, tell them their input was invalid. 2) When done playing the hand, repeat from step 1 """ n = HAND_SIZE hand= {} whileTrue: choice = input("Enter n to deal a new hand, r to replay the last hand, or e to end game: ") if choice == 'r': ifnot hand: print('You have not played a hand yet. Please play a new hand first!') else: playHand(hand, wordList, n) elif choice == 'n': hand = dealHand(n) playHand(hand, wordList, n) elif choice == 'e': break else: print("Invalid command.")
defcompChooseWord(hand, wordList, n): """ Given a hand and a wordList, find the word that gives the maximum value score, and return it. This word should be calculated by considering all the words in the wordList. If no words in the wordList can be made from the hand, return None. hand: dictionary (string -> int) wordList: list (string) n: integer (HAND_SIZE; i.e., hand size required for additional points) returns: string or None """ # Create a new variable to store the maximum score seen so far (initially 0) bestScore = 0 # Create a new variable to store the best word seen so far (initially None) bestWord = None # For each word in the wordList for word in wordList: # If you can construct the word from your hand if isValidWord(word, hand, wordList): # find out how much making that word is worth score = getWordScore(word, n) # If the score for that word is higher than your best score if (score > bestScore): # update your best score, and best word accordingly bestScore = score bestWord = word # return the best word you found return bestWord
defcompPlayHand(hand, wordList, n): """ Allows the computer to play the given hand, following the same procedure as playHand, except instead of the user choosing a word, the computer chooses it. 1) The hand is displayed. 2) The computer chooses a word. 3) After every valid word: the word and the score for that word is displayed, the remaining letters in the hand are displayed, and the computer chooses another word. 4) The sum of the word scores is displayed when the hand finishes. 5) The hand finishes when the computer has exhausted its possible choices (i.e. compChooseWord returns None). hand: dictionary (string -> int) wordList: list (string) n: integer (HAND_SIZE; i.e., hand size required for additional points) """ # Keep track of the total score totalScore = 0 # As long as there are still letters left in the hand: while (calculateHandlen(hand) > 0) : # Display the hand print("Current Hand: ", end=' ') displayHand(hand) # computer's word word = compChooseWord(hand, wordList, n) # If the input is a single period: if word == None: # End the game (break out of the loop) break # Otherwise (the input is not a single period): else : # If the word is not valid: if (not isValidWord(word, hand, wordList)) : print('This is a terrible error! I need to check my own code!') break # Otherwise (the word is valid): else : # Tell the user how many points the word earned, and the updated total score score = getWordScore(word, n) totalScore += score print('"' + word + '" earned ' + str(score) + ' points. Total: ' + str(totalScore) + ' points') # Update hand and show the updated hand to the user hand = updateHand(hand, word) print() # Game is over (user entered a '.' or ran out of letters), so tell user the total score print('Total score: ' + str(totalScore) + ' points.')
defplayGame(wordList): """ Allow the user to play an arbitrary number of hands. 1) Asks the user to input 'n' or 'r' or 'e'. * If the user inputs 'e', immediately exit the game. * If the user inputs anything that's not 'n', 'r', or 'e', keep asking them again. 2) Asks the user to input a 'u' or a 'c'. * If the user inputs anything that's not 'c' or 'u', keep asking them again. 3) Switch functionality based on the above choices: * If the user inputted 'n', play a new (random) hand. * Else, if the user inputted 'r', play the last hand again. * If the user inputted 'u', let the user play the game with the selected hand, using playHand. * If the user inputted 'c', let the computer play the game with the selected hand, using compPlayHand. 4) After the computer or user has played the hand, repeat from step 1 wordList: list (string) """ # TO DO... <-- Remove this comment when you code this function # print("playGame not yet implemented.") # <-- Remove this when you code this function n = HAND_SIZE hand= {} whileTrue: choice = input("Enter n to deal a new hand, r to replay the last hand, or e to end game: ") if choice == 'n': hand = dealHand(n) whileTrue: who = input("Enter u to have yourself play, c to have the computer play: ") if who notin ['u','c']: print('Invalid command.') else: break if who == 'u': playHand(hand, wordList, n) else: compPlayHand(hand, wordList, n) elif choice == 'e': break elif choice == 'r': ifnot hand: print('You have not played a hand yet. Please play a new hand first!') else: whileTrue: who = input("Enter u to have yourself play, c to have the computer play: ") if who notin ('u', 'c'): print('Invalid command.') else: break if who == 'u': playHand(hand, wordList, n) elif who == 'c': compPlayHand(hand, wordList, n) else: print("Invalid command.")
游戏运行
1 2 3
if __name__ == '__main__': wordList = loadWords() playGame(wordList)
Loading word list from file...
83667 words loaded.
Enter n to deal a new hand, r to replay the last hand, or e to end game: n
Enter u to have yourself play, c to have the computer play: c
Current Hand: u e w c x f n
"enuf" earned 28 points. Total: 28 points
Current Hand: w c x
Total score: 28 points.
Enter n to deal a new hand, r to replay the last hand, or e to end game: r
Enter u to have yourself play, c to have the computer play: u
Current Hand: u e w c x f n
Enter word, or a '.' to indicate that you are finished: 。
Invalid word, please try again.
Current Hand: u e w c x f n
Enter word, or a '.' to indicate that you are finished: .
Goodbye! Total score: 0 points.
Enter n to deal a new hand, r to replay the last hand, or e to end game: e
项目参考资料
John V. Guttag - Introduction to Computation and Programming Using Python_ With Application to Understanding Data (2016, The MIT Press)