One of the simplest two-player games is “Guess the number”. The first player thinks of a secret number in some known range while the second player attempts to guess the number. After each guess, the first player answers either “Higher”, “Lower” or “Correct!” depending on whether the secret number is higher, lower or equal to the guess. In this project, you will build a simple interactive program in Python where the computer will take the role of the first player while you play as the second player.
You will interact with your program using an input field and several buttons. For this project, we will ignore the canvas and print the computer’s responses in the console. Building an initial version of your project that prints information in the console is a development strategy that you should use in later projects as well. Focusing on getting the logic of the program correct before trying to make it display the information in some “nice” way on the canvas usually saves lots of time since debugging logic errors in graphical output can be tricky.
We have provided a basic template for this mini-project here. Our suggested development strategy for the basic version of “Guess the number” is below. Remember to run your program after each step to ensure that you implemented that step correctly.
Add code to the program template that creates a frame with an input field whose handler has the name input_guess
. You will use this input field to enter guesses.
Add code to the event handler input_guess(guess)
that takes the input string guess
, converts it to an integer, and prints out a message of the form "Guess was 37"
(or whatever the guess actually was). Hint: We have shown you how to convert strings to numbers in the lectures.
Add code to the function new_game
that initializes a global variable secret_number
to be a random number in the range [0, 100)
. Remember to include a global
statement. Hint: look at the functions in the random
module to figure out how to easily select such a random number. Note that the call to new_game()
at the bottom of the template ensures that secret_number
is always initialized when the program starts running.
When discussing ranges, we will follow the standard Python convention of including the low end of the range and excluding the high end of the range, which can be expressed mathematically as [low, high)
. So, [0, 3)
means all of the numbers starting at 0 up to, but not including 3. In other words 0, 1, and 2. We suggest using the range [0, 100)
in your first implementation.
Add code to input_guess
that compares the entered number to secret_number
and prints out an appropriate message such as "Higher"
, "Lower"
, or "Correct"
.
Test your code by playing multiple games of “Guess the number” with a fixed range. At this point, you will need to re-run your program between each game (using the CodeSkulptor “Run” button). You are also welcome to use this testing template http://www.codeskulptor.org/#examples-gtn_testing_template.py. The bottom of the template contains a sequence of calls to input_guess()
for three games of “Guess the number”. Uncomment each sequence of calls and check whether the output in the console matches that provided in the comments below. Note that your output doesn’t have to be identical, just of a similar form.
From this minimal working version of “Guess the number”, the rest of this project consists of adding extra functionality to your project. There are two improvements that you will need to make to get full credit:
simplegui
module, add buttons to restart the game so that you don’t need to repeatedly click “Run” in CodeSkulptor to play multiple games. You should add two buttons: “Range: 0 - 100” and “Range: 0 - 1000” that allow the player to choose different ranges for the secret number. Using either of these buttons should restart the game and print out an appropriate message. They should work at any time during the game. In our implementation, the event handler for each button set the desired range for the secret number (as a global variable) and then call new_game
to reset the secret number in the desired range.As you play “Guess the number”, you might notice that a good strategy is to maintain an interval that consists of the highest guess that is “Lower” than the secret number and the lowest guess that is “Higher” than the secret number. A good choice for the next guess is the number that is the average of these two numbers. The answer for this new guess then allows you to figure a new interval that contains the secret number and that is half as large. For example, if the secret number is in the range [0, 100)
, it is a good idea to guess 50
. If the answer is “Higher”, the secret number must be in the range [51, 100)
. It is then a good idea to guess 75
and so on. This technique of successively narrowing the range corresponds to a well-known computer algorithm known as binary search.
Since the strategy above for playing “Guess the number” approximately halves the range of possible secret numbers after each guess, any secret number in the range [low, high)
can always be found in at most n
guesses where n
is the smallest integer such that 2 ** n >= high - low + 1
. For the range [0, 100)
, n
is seven. For the range [0, 1000)
, n
is ten. In our implementation, the function new_game()
set the number of allowed guess to seven when the range is [0, 100)
or to ten when the range is [0, 1000)
. For more of a challenge, you may compute n
from low
and high
using the functions math.log
and math.ceil
in the math
module.
When your program starts, the game should immediately begin in range [0, 100)
. When the game ends (because the player either wins or runs out of guesses), a new game with the same range as the last one should immediately begin by calling new_game()
. Whenever the player clicks one of the range buttons, the current game should stop and a new game with the selected range should begin.
For more helpful tips on implementing this mini-project, please visit the Code Clinic tips page for this mini-project.
Your peers will assess your mini-project according to the rubric given below. To guide you in determining whether your project satisfies each item in the rubric, please consult the video that demonstrates our implementation of “Guess the number”. Small deviations from the textual output of our implementation are fine. You should avoid potentially confusing deviations (such as printing “Too high” or “Too low” instead of “Lower” and “Higher”). Whether moderate deviations satisfy an item of the grading rubric is at your peers’ discretion during their assessment.
Here is a break down of the scoring: - 1 pt – The game starts immediately when the “Run” button in CodeSkulptor is pressed. - 1 pt – A game is always in progress. Finishing one game immediately starts another in the same range. - 1 pt – The game reads guess
from the input field and correctly prints it out. - 3 pts – The game correctly plays “Guess the number” with the range [0, 100)
and prints understandable output messages to the console. Play three complete games: 1 pt for each correct game. - 2 pts – The game includes two buttons that allow the user to select the range [0, 100)
or the range [0, 1000)
for the secret number. These buttons correctly change the range and print an appropriate message. (1 pt per button.) - 2 pts – The game restricts the player to a finite number of guesses and correctly terminates the game when these guesses are exhausted. Award 1 pt if the number of remaining guesses is printed, but the game does not terminate correctly. - 1 pt – The game varies the number of allowed guesses based on the range of the secret number " seven guesses for range [0, 100)
, ten guesses for range [0, 1000)
.
To help aid you in gauging what a full credit project might look like, the video lecture on the “Guess the number” project includes a demonstration of our implementation of this project. You do not need to validate that the input number is in the correct range. (For this game, that responsibility should fall on the player.)
# "Guess the number"
# input will come from buttons and an input field
# all output for the game will be printed in the console
# online submission:
# http://www.codeskulptor.org/#user39_fDKDoZrKh7_2.py
import simplegui
import random
import math
range_min = 0
range_max = 100
remaining_guess = int(math.ceil(math.log(range_max - range_min + 1, 2)))
# helper function to start and restart the game
def new_game():
# initialize global variables used in your code here
global secret_number
secret_number = random.randrange(range_min, range_max)
global remaining_guess
remaining_guess = int(math.ceil(math.log(range_max - range_min + 1, 2)))
print " "
print "New game. Range is from", range_min, "to", range_max
print "Number of remaining guesses is", remaining_guess
# define event handlers for control panel
def range100():
# button that changes the range to [0,100) and starts a new game
global range_min, range_max
range_min = 0
range_max = 100
new_game()
def range1000():
# button that changes the range to [0,1000) and starts a new game
global range_min, range_max
range_min = 0
range_max = 1000
new_game()
def input_guess(guess):
# main game logic goes here
global remaining_guess
remaining_guess = remaining_guess - 1
guess = int(guess)
print " "
print "Guess was", guess
if secret_number > guess:
print "Number of remaining guesses is", remaining_guess
print "Higher!"
elif secret_number < guess:
print "Number of remaining guesses is", remaining_guess
print "Lower!"
else:
print "Number of remaining guesses is", remaining_guess
print "Correct!"
new_game()
if remaining_guess <= 0:
print "You ran out of guesses. The number was", secret_number
new_game()
# create frame
frame = simplegui.create_frame('Guess the number', 200, 150)
# register event handlers for control elements and start frame
frame.add_button('Range: [0, 100)', range100, 200)
frame.add_button('Range: [0, 1000)', range1000, 200)
frame.add_input('Enter your guess', input_guess, 200)
# call new_game
new_game()