This is one of my first programmes which I had written (definitely first over 100 lines) and I could really use some feedback.
import time
import getpass
import random
def invalid():
print("")
print("INVALID INPUT")
time.sleep(sec)
def play_game(secret,sec): #the actual game
def correct_guess(guess,secretword,output):
j=[]
t=-1 #Next couple lines is modefied code from https://stackoverflow.com/questions/20039022/python-finding-more-than-one-index-in-a-list-of-letters
while True:
try:
t=secretword.index(guess,t+1)
j.append(t)
except ValueError:
break
for z in j:
output[z]=guess
def draw_hangman(attempts): #generates image of hangman [attempts = number of wrong answers (up to 9)]
def legs(at):
if at==6:
if p==10:
print("* *")
print("* * *")
print("* *")
if at>=7:
if p==10:
print("* * *")
print("* * * *")
print("* *")
def rest(att):
if att in (1,2,3,4,5):
n={1: 2,2: 3,3: 5,4: 7,5: 10}
c=n[att]
if p in range(c,11):
print("*")
if p==11:
print("* *")
print("* *")
for p in range(1,13):
if p==1:
print("")
if attempts>=2:
if p==2:
print("* * * * * * * * *")
if attempts>=3:
if p in (3,4):
print("* *")
if attempts>=4:
if p==5:
print("* * *")
print("* *")
if attempts in (5,6,7):
if p in (7,8,9):
print("* *")
legs(attempts)
if attempts==8:
if p==7:
print("* * *")
print("* * *")
print("* *")
legs(attempts)
if attempts==9:
if p==7:
print("* * * *")
print("* * * *")
print("* *")
legs(attempts)
else:
rest(attempts)
else:
rest(attempts)
else:
rest(attempts)
else:
rest(attempts)
#secret = is the secret word / word = what the player already guessed
word=[]
for _ in secret:
word.append("_")
k=0
check=[] #here go all the guesses to make sure something isn't guessed twice
while True:
if secret==word:
print("")
print(word)
print("")
print("VICTORY")
break
continue
else:
print("")
print(word)
print("")
g=input("Enter Guess:")
if len(g)!=1:
invalid()
continue
if " " in g:
invalid()
continue
if g in check:
print("")
print("You already guessed this!")
time.sleep(sec)
continue
else:
check.append(g)
u=False
if g.lower() in secret:
correct_guess(g.lower(),secret,word)
u=True
if g.upper() in secret:
correct_guess(g.upper(),secret,word)
u=True
if u:
continue
else:
k=k+1
draw_hangman(k)
if k==9:
print("")
print(secret)
print("")
print("GAME OVER")
break
time.sleep(sec)
wordlibary= ['able', 'about', 'account', 'acid', 'across', 'act', 'addition', 'adjustment', 'advertisement', 'after', 'again', 'against', 'agreement', 'air', 'all', 'almost', 'among', 'amount', 'amusement', 'and', 'angle', 'angry', 'animal', 'answer', 'ant', 'any', 'apparatus', 'apple', 'approval', 'arch', 'argument', 'arm', 'army', 'art', 'as', 'at', 'attack', 'attempt', 'attention', 'attraction', 'authority', 'automatic', 'awake', 'baby', 'back', 'bad', 'bag', 'balance', 'ball', 'band', 'base', 'basin', 'basket', 'bath', 'be', 'beautiful', 'because', 'bed', 'bee', 'before', 'behaviour', 'belief', 'bell', 'bent', 'berry', 'between', 'bird', 'birth', 'bit', 'bite', 'bitter', 'black', 'blade', 'blood', 'blow', 'blue', 'board', 'boat', 'body', 'boiling', 'bone', 'book', 'boot', 'bottle', 'box', 'boy', 'brain', 'brake', 'branch', 'brass', 'bread', 'breath', 'brick', 'bridge', 'bright', 'broken', 'brother', 'brown', 'brush', 'bucket', 'building', 'bulb', 'burn', 'burst', 'business', 'but', 'butter', 'button', 'by', 'cake', 'camera', 'canvas', 'card', 'care', 'carriage', 'cart', 'cat', 'cause', 'certain', 'chain', 'chalk', 'chance', 'change', 'cheap', 'cheese', 'chemical', 'chest', 'chief', 'chin', 'church', 'circle', 'clean', 'clear', 'clock', 'cloth', 'cloud', 'coal', 'coat', 'cold', 'collar', 'colour', 'comb', 'come', 'comfort', 'committee', 'common', 'company', 'comparison', 'competition', 'complete', 'complex', 'condition', 'connection', 'conscious', 'control', 'cook', 'copper', 'copy', 'cord', 'cork', 'cotton', 'cough', 'country', 'cover', 'cow', 'crack', 'credit', 'crime', 'cruel', 'crush', 'cry', 'cup', 'cup', 'current', 'curtain', 'curve', 'cushion', 'damage', 'danger', 'dark', 'daughter', 'day', 'dead', 'dear', 'death', 'debt', 'decision', 'deep', 'degree', 'delicate', 'dependent', 'design', 'desire', 'destruction', 'detail', 'development', 'different', 'digestion', 'direction', 'dirty', 'discovery', 'discussion', 'disease', 'disgust', 'distance', 'distribution', 'division', 'do', 'dog', 'door', 'doubt', 'down', 'drain', 'drawer', 'dress', 'drink', 'driving', 'drop', 'dry', 'dust', 'ear', 'early', 'earth', 'east', 'edge', 'education', 'effect', 'egg', 'elastic', 'electric', 'end', 'engine', 'enough', 'equal', 'error', 'even', 'event', 'ever', 'every', 'example', 'exchange', 'existence', 'expansion', 'experience', 'expert', 'eye', 'face', 'fact', 'fall', 'false', 'family', 'far', 'farm', 'fat', 'father', 'fear', 'feather', 'feeble', 'feeling', 'female', 'fertile', 'fiction', 'field', 'fight', 'finger', 'fire', 'first', 'fish', 'fixed', 'flag', 'flame', 'flat', 'flight', 'floor', 'flower', 'fly', 'fold', 'food', 'foolish', 'foot', 'for', 'force', 'fork', 'form', 'forward', 'fowl', 'frame', 'free', 'frequent', 'friend', 'from', 'front', 'fruit', 'full', 'future', 'garden', 'general', 'get', 'girl', 'give', 'glass', 'glove', 'go', 'goat', 'gold', 'good', 'government', 'grain', 'grass', 'great', 'green', 'grey', 'grip', 'group', 'growth', 'guide', 'gun', 'hair', 'hammer', 'hand', 'hanging', 'happy', 'harbour', 'hard', 'harmony', 'hat', 'hate', 'have', 'he', 'head', 'healthy', 'hear', 'hearing', 'heart', 'heat', 'help', 'high', 'history', 'hole', 'hollow', 'hook', 'hope', 'horn', 'horse', 'hospital', 'hour', 'house', 'how', 'humour', 'I', 'ice', 'idea', 'if', 'ill', 'important', 'impulse', 'in', 'increase', 'industry', 'ink', 'insect', 'instrument', 'insurance', 'interest', 'invention', 'iron', 'island', 'jelly', 'jewel', 'join', 'journey', 'judge', 'jump', 'keep', 'kettle', 'key', 'kick', 'kind', 'kiss', 'knee', 'knife', 'knot', 'knowledge', 'land', 'language', 'last', 'late', 'laugh', 'law', 'lead', 'leaf', 'learning', 'leather', 'left', 'leg', 'let', 'letter', 'level', 'library', 'lift', 'light', 'like', 'limit', 'line', 'linen', 'lip', 'liquid', 'list', 'little', 'living', 'lock', 'long', 'look', 'loose', 'loss', 'loud', 'love', 'low', 'machine', 'make', 'male', 'man', 'manager', 'map', 'mark', 'market', 'married', 'mass', 'match', 'material', 'may', 'meal', 'measure', 'meat', 'medical', 'meeting', 'memory', 'metal', 'middle', 'military', 'milk', 'mind', 'mine', 'minute', 'mist', 'mixed', 'money', 'monkey', 'month', 'moon', 'morning', 'mother', 'motion', 'mountain', 'mouth', 'move', 'much', 'muscle', 'music', 'nail', 'name', 'narrow', 'nation', 'natural', 'near', 'necessary', 'neck', 'need', 'needle', 'nerve', 'net', 'new', 'news', 'night', 'no', 'noise', 'normal', 'north', 'nose', 'not', 'note', 'now', 'number', 'nut', 'observation', 'of', 'off', 'offer', 'office', 'oil', 'old', 'on', 'only', 'open', 'operation', 'opinion', 'opposite', 'or', 'orange', 'order', 'organization', 'ornament', 'other', 'out', 'oven', 'over', 'owner', 'page', 'pain', 'paint', 'paper', 'parallel', 'parcel', 'part', 'past', 'paste', 'payment', 'peace', 'pen', 'pencil', 'person', 'physical', 'picture', 'pig', 'pin', 'pipe', 'place', 'plane', 'plant', 'plate', 'play', 'please', 'pleasure', 'plough', 'pocket', 'point', 'poison', 'polish', 'political', 'poor', 'porter', 'position', 'possible', 'pot', 'potato', 'powder', 'power', 'present', 'price', 'print', 'prison', 'private', 'probable', 'process', 'produce', 'profit', 'property', 'prose', 'protest', 'public', 'pull', 'pump', 'punishment', 'purpose', 'push', 'put', 'quality', 'question', 'quick', 'quiet', 'quite', 'rail', 'rain', 'range', 'rat', 'rate', 'ray', 'reaction', 'reading', 'ready', 'reason', 'receipt', 'record', 'red', 'regret', 'regular', 'relation', 'religion', 'representative', 'request', 'respect', 'responsible', 'rest', 'reward', 'rhythm', 'rice', 'right', 'ring', 'river', 'road', 'rod', 'roll', 'roof', 'room', 'root', 'rough', 'round', 'rub', 'rule', 'run', 'sad', 'safe', 'sail', 'salt', 'same', 'sand', 'say', 'scale', 'school', 'science', 'scissors', 'screw', 'sea', 'seat', 'second', 'secret', 'secretary', 'see', 'seed', 'seem', 'selection', 'self', 'send', 'sense', 'separate', 'serious', 'servant', 'sex', 'shade', 'shake', 'shame', 'sharp', 'sheep', 'shelf', 'ship', 'shirt', 'shock', 'shoe', 'short', 'shut', 'side', 'sign', 'silk', 'silver', 'simple', 'sister', 'size', 'skin', '', 'skirt', 'sky', 'sleep', 'slip', 'slope', 'slow', 'small', 'smash', 'smell', 'smile', 'smoke', 'smooth', 'snake', 'sneeze', 'snow', 'so', 'soap', 'society', 'sock', 'soft', 'solid', 'some', '', 'son', 'song', 'sort', 'sound', 'soup', 'south', 'space', 'spade', 'special', 'sponge', 'spoon', 'spring', 'square', 'stage', 'stamp', 'star', 'start', 'statement', 'station', 'steam', 'steel', 'stem', 'step', 'stick', 'sticky', 'stiff', 'still', 'stitch', 'stocking', 'stomach', 'stone', 'stop', 'store', 'story', 'straight', 'strange', 'street', 'stretch', 'strong', 'structure', 'substance', 'such', 'sudden', 'sugar', 'suggestion', 'summer', 'sun', 'support', 'surprise', 'sweet', 'swim', 'system', 'table', 'tail', 'take', 'talk', 'tall', 'taste', 'tax', 'teaching', 'tendency', 'test', 'than', 'that', 'the', 'then', 'theory', 'there', 'thick', 'thin', 'thing', 'this', 'thought', 'thread', 'throat', 'through', 'through', 'thumb', 'thunder', 'ticket', 'tight', 'till', 'time', 'tin', 'tired', 'to', 'toe', 'together', 'tomorrow', 'tongue', 'tooth', 'top', 'touch', 'town', 'trade', 'train', 'transport', 'tray', 'tree', 'trick', 'trouble', 'trousers', 'true', 'turn', 'twist', 'umbrella', 'under', 'unit', 'up', 'use', 'value', 'verse', 'very', 'vessel', 'view', 'violent', 'voice', 'waiting', 'walk', 'wall', 'war', 'warm', 'wash', 'waste', 'watch', 'water', 'wave', 'wax', 'way', 'weather', 'week', 'weight', 'well', 'west', 'wet', 'wheel', 'when', 'where', 'while', 'whip', 'whistle', 'white', 'who', 'why', 'wide', 'will', 'wind', 'window', 'wine', 'wing', 'winter', 'wire', 'wise', 'with', 'woman', 'wood', 'wool', 'word', 'work', 'worm', 'wound', 'writing', 'wrong', 'year', 'yellow', 'yes', 'yesterday', 'you', 'young', 'Bernhard', 'Breytenbach', 'Android']
# got this list from https://github.com/Xethron/Hangman/blob/master/words.txt
sec=0.8 #since all time delays are the same I wanted them all to be a variable so it is easeier to change it
while True:
print("")
print("1:New Game(Random Word)")
print("2:New Game(With your own Word)")
print("3:Exit Game")
x=input("Enter(1,2,3):")
if x=="2":
while True:
print("")
wordchoice=getpass.getpass("Enter secret word:")
y=False
if " " in wordchoice:
print("")
print("Only one word!")
time.sleep(sec)
continue
if len(wordchoice)<1:
invalid()
continue
alphabet="abcdefghijklmnopqrstuvwxyz"
wordlist=list(wordchoice)
for n in wordlist:
if n in alphabet.lower():
continue
if n in alphabet.upper():
continue
else:
print("")
print("Use only letters!")
time.sleep(sec)
y=True
break
if y:
continue
else:
play_game(wordlist,sec)
break
continue
if x=="1":
wordlist=list(random.choice(wordlibary))
play_game(wordlist,sec)
continue
if x=="3":
break
else:
invalid()
-
\$\begingroup\$ Generally, we prefer you to keep your original code in the question; this allows others to see the process. If you want the new code to be reviewed, you can ask a new question. In any case, please revert your most recent edit. \$\endgroup\$Graham– Graham2018年11月17日 23:30:22 +00:00Commented Nov 17, 2018 at 23:30
-
1\$\begingroup\$ Adding a revised version of your code is also discouraged, per this meta Code Review answer ("You also should not append your revised code to the question"), however I think I can show you a few more things on my answer before there's a need to ask a new question. \$\endgroup\$Graham– Graham2018年11月17日 23:49:36 +00:00Commented Nov 17, 2018 at 23:49
-
\$\begingroup\$ Even if you want to learn by doing (which is a good philosophy), I would recommend going back and comparing your code to my code line by line afterwards. Figure out why I did something a certain way. If your code is longer, then it's almost certainly less clear (although that's not necessarily bad in the general case if it's a necessary optimization, but that will not be the case for this code.) If you can't figure out why I did something, leave a comment. If you want to hone the code down even further (which is certainly possible; I improved it, but there's other things), ask a new question. \$\endgroup\$Graham– Graham2018年11月18日 00:10:08 +00:00Commented Nov 18, 2018 at 0:10
-
1\$\begingroup\$ I would just recommend making sure you understand my code entirely, and have good reason for any deviations before asking a new question. \$\endgroup\$Graham– Graham2018年11月18日 00:11:20 +00:00Commented Nov 18, 2018 at 0:11
-
2\$\begingroup\$ Welcome to Code Review. I have rolled back your edits. Please do not update the code in your question to incorporate feedback from answers, doing so goes against the Question + Answer style of Code Review. This is not a forum where you should keep the most updated version in your question. Please see what you may and may not do after receiving answers . \$\endgroup\$Heslacher– Heslacher2018年11月19日 06:09:09 +00:00Commented Nov 19, 2018 at 6:09
1 Answer 1
Read this entire section first
Welcome to Code Review! I enjoyed trying out your game and playing it. Don't be discouraged by my long list of improvements; there's always a lot to learn when you're just starting, and every new thing you try brings its own challenges and best practices.
I would recommend reading the entire list and trying to understand the purpose of each of the suggestions. I have provided a revised version of your code at the end of my answer. The list is not ordered by importance, just by my flow when I was reviewing your code. Hopefully the logical order makes sense, and you'll be able to use this answer as a reference.
Observe PEP-8 standards
PEP-8 is the widely-accepted style guide for Python and it makes Python code easier to read and more standard. It's basically the common language of Python programmers. I would recommend beginning to read it and learning the different guidelines. Here are a few style suggestions (The small text in parenthesis is regex replacements. You probably don't need to worry about them; they're just my quick way of fixing code in a text editor. For those who are familiar with regex
, I always preview all the changes before applying the regex
.):
- I notice you often do not put spaces on both sides of your
=
and comparison operators (==
,!=
,>=
, and<=
). PEP-8 recommends putting spaces on both sides would make your code easier to read. (I fixed this using the regex?([><!]?=+) ?
and replacing with1円
.) - I would also recommend using more descriptive variable names in certain places throughout your code: on first glance, I'm not sure what
j
,t
, andz
represent. - Comments should generally be on a separate line from code, and avoided by using more clear code where possible (I'm referring to the comment after the
draw_hangman
declaration.) - Almost universally, you should have spaces after commas. Sometimes it's readable in function argument lists, but for declaring a
dict
orlist
, it makes the code much more readable. (This is easy to fix using regex:,(?! )
to,
)
Nested methods
I notice you have a method hierarchy:
def play_game():
# ...
def correct_guess()
# ...
def draw_hangman()
def legs():
# ...
def rest():
# ...
Unless you want to restrict the scope of methods for a good reason, you shouldn't use nested methods. Just lay them flat on the same level in your file. If this makes the purpose of a method unclear, rename it and utilize docstrings. However, I think you partly did this to take advantage of being able to reference the function-scoped variable within another function. Instead, just pass the required variables as function arguments to the previous subfunction, and add those parameter the previous subfunction. For example, you reference p
in legs()
and rest()
: just add p
as a parameter to legs()
and rest()
.
Encoding your hangman graphic
Let's talk about draw_hangman()
:
def draw_hangman(attempts):
def legs(at):
# ...
def rest(att):
# ...
for p in range(1, 13):
if p == 1:
print("")
if attempts >= 2:
if p == 2:
print("* * * * * * * * *")
if attempts >= 3:
if p in (3, 4):
print("* *")
if attempts >= 4:
# ...
else:
rest(attempts)
else:
rest(attempts)
else:
rest(attempts)
else:
rest(attempts)
Firstly, your for p in range(1, 13)
and generally the existence of the p
variable is a bad pattern. All you're doing is printing out a set of lines in order; checking for where p
is just makes your code unnecessarily inefficient and confusing to follow. As a general rule, you should never iterate through a range
and then check your current position in the range to create conditionals; you should just do the things you need to do in order, and make conditional checks within at the appropriate points to change the output for specific parts. Refactoring this is complicated, but fortunately it will be unnecessary, as we'll get to in my next paragraph.
Instead of using conditionals to determine how to print the picture, you could simply encode the picture in an iterative way and then retrieve it conditionally. Let me show you what I mean; you can start with original hangman graphic:
* * * * * * * * *
* *
* *
* * *
* *
* * * *
* * * *
* *
* * *
* * * *
* *
and then number it corresponding to when specific parts should be displayed:
1 2 2 2 2 2 2 2 2
1 3
1 3
1 4 4
1 4
1 8 5 9
1 8 5 9
1 5
1 6 7
1 1 6 7
1 1
Conveniently, your hangman happens to have few enough parts to allow for string storage to with digits 1-9 to work; otherwise you would have to store the hangman image in a list.
You can then retrieve the image using string replacement. The python regex module re
is particularly convenient, as I will demonstrate in the following example:
import re
# ...
hangman_image = """
1 2 2 2 2 2 2 2 2
1 3
1 3
1 4 4
1 4
1 8 5 9
1 8 5 9
1 5
1 6 7
1 1 6 7
1 1"""
# generates image of hangman [attempts = number of wrong answers (up to 9)]
def draw_hangman(attempts):
current_hangman = hangman_image
current_hangman = re.sub('[1-{}]'.format(attempts), '*', current_hangman)
current_hangman = re.sub('[{}-9]'.format(attempts + 1), ' ', current_hangman)
print(current_hangman)
The first time re.sub()
is called, the regular expression replaces all numbers in the range 1 to attempts
. The value of attempts is inserted into the string [1-{}]
by using .format()
to replace {}
with the value of attempts
. In regular expressions, brackets surrounding a concatenation of characters form what is called a "character class": this just means it's looking for a single character inside the brackets and then replacing it. Character classes support ranges of characters (relying on the way characters are encoded) by using characters separated by a hyphen for both ends of the range. The second time re.sub()
is called is similar, but it instead targets all the numbers which are greater than the number of attempts and replaces them with spaces.
This conveniently also eliminates the need for the legs()
and rest()
functions.
Main loop
To start with, kudos for making sec
a variable. Generally, when you have repeated constants you should make them variables. Even if they aren't repeated, making them a variable can make procedured more clear (see Magic constant / Magic number).
Your main loop and your code generally needs more clear control flow. Instead of having multiple if
clauses for mutually exclusive outcomes, you can use elif
and else
to direct mutually exclusive outcomes.
if x == "2":
# ...
elif x == "1":
# ...
elif x == "3":
break
Then, you need to clean up all your continues
and breaks
; an easy immediate change is to remove the continue
s and break
s preceding the each x
comparison conditional. As you're using them now, they are a form of goto
, and as Edgar Dijkstra famously noted in a paper, Go To Statement Considered Harmful
(this is not to say all continue
s and break
s are bad, but you should probably minimize their use when possible.) Generally, you should use else
more to obviate the need to use continue
s and break
s.
I will describe the refactoring process below:
The main revisions are inside if x == 2:
- Made the if statements checking the
wordchoice
into a chain ofif elif
statements, and then moved the remaining code in thewhile True
loop into anelse
clause. Thecontinue
s within the first two conditionals can now be removed. - Python provides the lowercase and uppercase Roman alphabet (A.K.A. the English alphabet) as an easy import: just add
from string import ascii_letters
; however this is not even necessary... - EDIT: As you correctly pointed out, my original regex method (
\w+
) didn't exclude numbers. To check if a word contains just ascii characters, just use the follow regular expression:[a-zA-Z]+
. You can compare a string usingre.fullmatch
. Refactoring the code to use this will eliminate the need for they
variable. - In the
invalid()
function, it's unclear wheresec
comes from.sec
should passed as a parameter toinvalid()
. It's poor form to reference a global variable where the origin of the variable is unclear. - As a matter of simplicity, just using a list of lowercase words will simplify things, since you don't check for case inside the
play_game()
; moving the list conversion of the word insideplay_game()
would also be a good idea to separate abstraction.
Improving playgame()
Here, the problems are similar to the problems with your main loop; your control flow is very unclear. Your variable names could also use some work.
- A list comprehension can allow you to declare
word
in one line:word = ["_" for _ in secret]
- A
continue
directly preceded by abreak
does nothing. Thecontinue
will never be reached because thebreak
will change the control flow. - You should rename:
k
towrong_guesses
check
toguessed_letters
g
toguess
.
- Multiple consecutive conditionals should never all have
continue
s within them. Just useelse
andelif
; you can useor
to join two conditions that have the same result. - Instead of doing
wrong_guesses = wrong_guesses + 1
, you can increment using+=
likewrong_guesses += 1
.
I would get rid of correct_guess
, because you're trying to rely on list mutation to modify a variable. Instead of passing a list and mutating it, you should just do the changes inline as follows (this is also coupled with a change to make secret
all lowercase in the beginning of playgame
):
if guess.lower() in secret or guess.upper() in secret:
for i in range(len(secret)):
if guess == secret[i]:
word[i] = guess
Final code
I've tested the following code and it works! It can hopefully illustrate how to do a better form of control flow and modularization.
import time
import getpass
import random
import re
def invalid(sec):
print("")
print("INVALID INPUT")
time.sleep(sec)
hangman_image = """
1 2 2 2 2 2 2 2 2
1 3
1 3
1 4 4
1 4
1 8 5 9
1 8 5 9
1 5
1 6 7
1 1 6 7
1 1"""
# generates image of hangman [attempts = number of wrong answers (up to 9)]
def draw_hangman(attempts):
current_hangman = hangman_image
current_hangman = re.sub('[1-{}]'.format(attempts), '*', current_hangman)
current_hangman = re.sub('[{}-9]'.format(attempts + 1), ' ', current_hangman)
print(current_hangman)
def play_game(target_word, sec): #the actual game
secret = list(target_word.lower())
word = ["_" for _ in secret]
wrong_guesses = 0
guessed_letters = []
while True:
print()
print(word)
print()
guess = None
while True:
guess = input("Enter Guess:")
if len(guess) != 1 or " " in guess:
invalid(sec)
elif guess in guessed_letters:
print()
print("You already guessed this!")
time.sleep(sec)
else:
break
guessed_letters.append(guess)
if guess.lower() in secret or guess.upper() in secret:
for i in range(len(secret)):
if guess == secret[i]:
word[i] = guess
else:
wrong_guesses = wrong_guesses + 1
draw_hangman(wrong_guesses)
if wrong_guesses == 9:
print()
print(secret)
print()
print("GAME OVER")
break
time.sleep(sec)
if word == secret:
print()
print(target_word)
print()
print("VICTORY")
return
# got this list from https://github.com/Xethron/Hangman/blob/master/words.txt
wordlibrary = ['able', 'about', 'account', 'acid', 'across', 'act', 'addition', 'adjustment', 'advertisement', 'after', 'again', 'against', 'agreement', 'air', 'all', 'almost', 'among', 'amount', 'amusement', 'and', 'angle', 'angry', 'animal', 'answer', 'ant', 'any', 'apparatus', 'apple', 'approval', 'arch', 'argument', 'arm', 'army', 'art', 'as', 'at', 'attack', 'attempt', 'attention', 'attraction', 'authority', 'automatic', 'awake', 'baby', 'back', 'bad', 'bag', 'balance', 'ball', 'band', 'base', 'basin', 'basket', 'bath', 'be', 'beautiful', 'because', 'bed', 'bee', 'before', 'behaviour', 'belief', 'bell', 'bent', 'berry', 'between', 'bird', 'birth', 'bit', 'bite', 'bitter', 'black', 'blade', 'blood', 'blow', 'blue', 'board', 'boat', 'body', 'boiling', 'bone', 'book', 'boot', 'bottle', 'box', 'boy', 'brain', 'brake', 'branch', 'brass', 'bread', 'breath', 'brick', 'bridge', 'bright', 'broken', 'brother', 'brown', 'brush', 'bucket', 'building', 'bulb', 'burn', 'burst', 'business', 'but', 'butter', 'button', 'by', 'cake', 'camera', 'canvas', 'card', 'care', 'carriage', 'cart', 'cat', 'cause', 'certain', 'chain', 'chalk', 'chance', 'change', 'cheap', 'cheese', 'chemical', 'chest', 'chief', 'chin', 'church', 'circle', 'clean', 'clear', 'clock', 'cloth', 'cloud', 'coal', 'coat', 'cold', 'collar', 'colour', 'comb', 'come', 'comfort', 'committee', 'common', 'company', 'comparison', 'competition', 'complete', 'complex', 'condition', 'connection', 'conscious', 'control', 'cook', 'copper', 'copy', 'cord', 'cork', 'cotton', 'cough', 'country', 'cover', 'cow', 'crack', 'credit', 'crime', 'cruel', 'crush', 'cry', 'cup', 'cup', 'current', 'curtain', 'curve', 'cushion', 'damage', 'danger', 'dark', 'daughter', 'day', 'dead', 'dear', 'death', 'debt', 'decision', 'deep', 'degree', 'delicate', 'dependent', 'design', 'desire', 'destruction', 'detail', 'development', 'different', 'digestion', 'direction', 'dirty', 'discovery', 'discussion', 'disease', 'disgust', 'distance', 'distribution', 'division', 'do', 'dog', 'door', 'doubt', 'down', 'drain', 'drawer', 'dress', 'drink', 'driving', 'drop', 'dry', 'dust', 'ear', 'early', 'earth', 'east', 'edge', 'education', 'effect', 'egg', 'elastic', 'electric', 'end', 'engine', 'enough', 'equal', 'error', 'even', 'event', 'ever', 'every', 'example', 'exchange', 'existence', 'expansion', 'experience', 'expert', 'eye', 'face', 'fact', 'fall', 'false', 'family', 'far', 'farm', 'fat', 'father', 'fear', 'feather', 'feeble', 'feeling', 'female', 'fertile', 'fiction', 'field', 'fight', 'finger', 'fire', 'first', 'fish', 'fixed', 'flag', 'flame', 'flat', 'flight', 'floor', 'flower', 'fly', 'fold', 'food', 'foolish', 'foot', 'for', 'force', 'fork', 'form', 'forward', 'fowl', 'frame', 'free', 'frequent', 'friend', 'from', 'front', 'fruit', 'full', 'future', 'garden', 'general', 'get', 'girl', 'give', 'glass', 'glove', 'go', 'goat', 'gold', 'good', 'government', 'grain', 'grass', 'great', 'green', 'grey', 'grip', 'group', 'growth', 'guide', 'gun', 'hair', 'hammer', 'hand', 'hanging', 'happy', 'harbour', 'hard', 'harmony', 'hat', 'hate', 'have', 'he', 'head', 'healthy', 'hear', 'hearing', 'heart', 'heat', 'help', 'high', 'history', 'hole', 'hollow', 'hook', 'hope', 'horn', 'horse', 'hospital', 'hour', 'house', 'how', 'humour', 'I', 'ice', 'idea', 'if', 'ill', 'important', 'impulse', 'in', 'increase', 'industry', 'ink', 'insect', 'instrument', 'insurance', 'interest', 'invention', 'iron', 'island', 'jelly', 'jewel', 'join', 'journey', 'judge', 'jump', 'keep', 'kettle', 'key', 'kick', 'kind', 'kiss', 'knee', 'knife', 'knot', 'knowledge', 'land', 'language', 'last', 'late', 'laugh', 'law', 'lead', 'leaf', 'learning', 'leather', 'left', 'leg', 'let', 'letter', 'level', 'library', 'lift', 'light', 'like', 'limit', 'line', 'linen', 'lip', 'liquid', 'list', 'little', 'living', 'lock', 'long', 'look', 'loose', 'loss', 'loud', 'love', 'low', 'machine', 'make', 'male', 'man', 'manager', 'map', 'mark', 'market', 'married', 'mass', 'match', 'material', 'may', 'meal', 'measure', 'meat', 'medical', 'meeting', 'memory', 'metal', 'middle', 'military', 'milk', 'mind', 'mine', 'minute', 'mist', 'mixed', 'money', 'monkey', 'month', 'moon', 'morning', 'mother', 'motion', 'mountain', 'mouth', 'move', 'much', 'muscle', 'music', 'nail', 'name', 'narrow', 'nation', 'natural', 'near', 'necessary', 'neck', 'need', 'needle', 'nerve', 'net', 'new', 'news', 'night', 'no', 'noise', 'normal', 'north', 'nose', 'not', 'note', 'now', 'number', 'nut', 'observation', 'of', 'off', 'offer', 'office', 'oil', 'old', 'on', 'only', 'open', 'operation', 'opinion', 'opposite', 'or', 'orange', 'order', 'organization', 'ornament', 'other', 'out', 'oven', 'over', 'owner', 'page', 'pain', 'paint', 'paper', 'parallel', 'parcel', 'part', 'past', 'paste', 'payment', 'peace', 'pen', 'pencil', 'person', 'physical', 'picture', 'pig', 'pin', 'pipe', 'place', 'plane', 'plant', 'plate', 'play', 'please', 'pleasure', 'plough', 'pocket', 'point', 'poison', 'polish', 'political', 'poor', 'porter', 'position', 'possible', 'pot', 'potato', 'powder', 'power', 'present', 'price', 'print', 'prison', 'private', 'probable', 'process', 'produce', 'profit', 'property', 'prose', 'protest', 'public', 'pull', 'pump', 'punishment', 'purpose', 'push', 'put', 'quality', 'question', 'quick', 'quiet', 'quite', 'rail', 'rain', 'range', 'rat', 'rate', 'ray', 'reaction', 'reading', 'ready', 'reason', 'receipt', 'record', 'red', 'regret', 'regular', 'relation', 'religion', 'representative', 'request', 'respect', 'responsible', 'rest', 'reward', 'rhythm', 'rice', 'right', 'ring', 'river', 'road', 'rod', 'roll', 'roof', 'room', 'root', 'rough', 'round', 'rub', 'rule', 'run', 'sad', 'safe', 'sail', 'salt', 'same', 'sand', 'say', 'scale', 'school', 'science', 'scissors', 'screw', 'sea', 'seat', 'second', 'secret', 'secretary', 'see', 'seed', 'seem', 'selection', 'self', 'send', 'sense', 'separate', 'serious', 'servant', 'sex', 'shade', 'shake', 'shame', 'sharp', 'sheep', 'shelf', 'ship', 'shirt', 'shock', 'shoe', 'short', 'shut', 'side', 'sign', 'silk', 'silver', 'simple', 'sister', 'size', 'skin', '', 'skirt', 'sky', 'sleep', 'slip', 'slope', 'slow', 'small', 'smash', 'smell', 'smile', 'smoke', 'smooth', 'snake', 'sneeze', 'snow', 'so', 'soap', 'society', 'sock', 'soft', 'solid', 'some', '', 'son', 'song', 'sort', 'sound', 'soup', 'south', 'space', 'spade', 'special', 'sponge', 'spoon', 'spring', 'square', 'stage', 'stamp', 'star', 'start', 'statement', 'station', 'steam', 'steel', 'stem', 'step', 'stick', 'sticky', 'stiff', 'still', 'stitch', 'stocking', 'stomach', 'stone', 'stop', 'store', 'story', 'straight', 'strange', 'street', 'stretch', 'strong', 'structure', 'substance', 'such', 'sudden', 'sugar', 'suggestion', 'summer', 'sun', 'support', 'surprise', 'sweet', 'swim', 'system', 'table', 'tail', 'take', 'talk', 'tall', 'taste', 'tax', 'teaching', 'tendency', 'test', 'than', 'that', 'the', 'then', 'theory', 'there', 'thick', 'thin', 'thing', 'this', 'thought', 'thread', 'throat', 'through', 'through', 'thumb', 'thunder', 'ticket', 'tight', 'till', 'time', 'tin', 'tired', 'to', 'toe', 'together', 'tomorrow', 'tongue', 'tooth', 'top', 'touch', 'town', 'trade', 'train', 'transport', 'tray', 'tree', 'trick', 'trouble', 'trousers', 'true', 'turn', 'twist', 'umbrella', 'under', 'unit', 'up', 'use', 'value', 'verse', 'very', 'vessel', 'view', 'violent', 'voice', 'waiting', 'walk', 'wall', 'war', 'warm', 'wash', 'waste', 'watch', 'water', 'wave', 'wax', 'way', 'weather', 'week', 'weight', 'well', 'west', 'wet', 'wheel', 'when', 'where', 'while', 'whip', 'whistle', 'white', 'who', 'why', 'wide', 'will', 'wind', 'window', 'wine', 'wing', 'winter', 'wire', 'wise', 'with', 'woman', 'wood', 'wool', 'word', 'work', 'worm', 'wound', 'writing', 'wrong', 'year', 'yellow', 'yes', 'yesterday', 'you', 'young', 'Bernhard', 'Breytenbach', 'Android']
# since all time delays are the same I wanted them all to be a variable so it is easier to change it
sec = 0.8
while True:
print()
print("1:New Game(Random Word)")
print("2:New Game(With your own Word)")
print("3:Exit Game")
x = input("Enter(1, 2, 3):")
if x == "2":
wordchoice = None
while True:
print()
wordchoice = getpass.getpass("Enter secret word:")
print()
if " " in wordchoice:
print("Only one word!")
time.sleep(sec)
elif len(wordchoice) < 1:
print("Enter more than zero characters.")
elif not re.fullmatch('[a-zA-Z]+', wordchoice):
print("Use only letters!")
else:
break
time.sleep(sec)
play_game(wordchoice, sec)
elif x == "1":
play_game(random.choice(wordlibrary), sec)
elif x == "3":
break
else:
invalid(sec)
Miscellaneous notes
- I would move
wordlibrary
to a separate file and then import it. Or read it from a file. Also, you misspelled "wordlibrary" as "wordlibary".- I suggested this because it takes up a lot of space in your current file, and it's not something you need to look at when you're editing the code of the file; it's really an entirely separate component.
- If you're curious, I could also tell you a way to make your
print()
statements more succinct, but I didn't want to get into it in the main review.- The easiest way to make
print()
statements more succint is to take advantage of newline characters and formatted string literals. For example if you wanted to turn the three linesprint()
,print(secret)
,print()
into one line, you would do:print(f'\n{secret}\n')
- The easiest way to make
-
1\$\begingroup\$ Thank you for your feedback! I implemented most of your suggestions into my code, however, didn't copy your final code since I wanted to practice using your final code only as a reference when necessary. Something worth mentioning is that (elif not re.fullmatch('\w+', wordchoice):) doesn't seem to work when I implement it in my code or if I run yours because when I tried entering numbers it didn't print use only letters. Couldn't figure out why. Also what would be the advantages of wordlibrary being a seperate file? And yes I am curious how I could make my print() succinct. Thanks for feedback \$\endgroup\$Jonas– Jonas2018年11月17日 23:44:55 +00:00Commented Nov 17, 2018 at 23:44
-
\$\begingroup\$ You're correct, I accidentally made a mistake with the regex. You can fix the regex problem by replacing
\w+
with[a-zA-z]+
; I've edited that into my answer, along with explanations for the other two parts. \$\endgroup\$Graham– Graham2018年11月18日 00:01:26 +00:00Commented Nov 18, 2018 at 0:01