Pokemon Turn Based battle (Python) The 2019 Stack Overflow Developer Survey Results Are InTurn-based battle simulatorBattle simulator RPGText-based turn-based fighter gamePokemon battle simulatorPokemon-style text battle gameDiscount Pokemon BattlesExurbb (automated turn based) Battle SimulatorExurbb (automated turn based) Battle Simulator RevisedCalculating the percentage on Python 2.7Text-Turn based dueling game
Why hard-Brexiteers don't insist on a hard border to prevent illegal immigration after Brexit?
Should I use my personal e-mail address, or my workplace one, when registering to external websites for work purposes?
Why was M87 targetted for the Event Horizon Telescope instead of Sagittarius A*?
Is a "Democratic" Oligarchy-Style System Possible?
What is the meaning of the verb "bear" in this context?
If a Druid sees an animal’s corpse, can they Wild Shape into that animal?
Did Section 31 appear in Star Trek: The Next Generation?
How technical should a Scrum Master be to effectively remove impediments?
Right tool to dig six foot holes?
What is the meaning of Triage in Cybersec world?
Deal with toxic manager when you can't quit
Geography at the pixel level
Can one be advised by a professor who is very far away?
Shouldn't "much" here be used instead of "more"?
Button changing it's text & action. Good or terrible?
Why didn't the Event Horizon Telescope team mention Sagittarius A*?
Worn-tile Scrabble
Why do UK politicians seemingly ignore opinion polls on Brexit?
What do hard-Brexiteers want with respect to the Irish border?
Aging parents with no investments
What tool would a Roman-age civilization have for the breaking of silver and other metals into dust?
Does the shape of a die affect the probability of a number being rolled?
Why not take a picture of a closer black hole?
Feature engineering suggestion required
Pokemon Turn Based battle (Python)
The 2019 Stack Overflow Developer Survey Results Are InTurn-based battle simulatorBattle simulator RPGText-based turn-based fighter gamePokemon battle simulatorPokemon-style text battle gameDiscount Pokemon BattlesExurbb (automated turn based) Battle SimulatorExurbb (automated turn based) Battle Simulator RevisedCalculating the percentage on Python 2.7Text-Turn based dueling game
.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty margin-bottom:0;
$begingroup$
This is based off the popular Pokemon Turn Based project (here).
GOAL
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
SUBGOALS
When someone is defeated, make sure the game prints out that their health has reached 0, and not a negative number.
When the computer's health reaches a set amount (such as 35%), increase it's chance to cast heal.
Give each move a name.
I am new to Python and I am trying to use OOP. Did I use my classes properly and is there a way to reduce the number of if statements in my code? Since this is my first code review, how is the formatting, number of comments, etc?
import random
class Pokemon:
"""Blueprint for turn based Pokemon battle"""
def __init__(self, attack_choice):
self.__attack_choice = attack_choice
def attack(self):
if self.__attack_choice == 1:
attack_points = random.randint(18,25)
return attack_points
elif self.__attack_choice == 2:
attack_points = random.randint(10,35)
return attack_points
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18,35)
return heal_points
###########################################################################
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = eval(input("nSelect an attack: "))
# Mew selects an attack, but focuses on attacking if health is full.
if mew_health == 100:
mew_choice = random.randint(1,2)
else:
mew_choice = random.randint(1,3)
mew = Pokemon(mew_choice)
user_pokemon = Pokemon(attack_choice)
# Attacks by user and Mew are done simultaneously.
if attack_choice == 1 or attack_choice == 2:
damage_to_mew = user_pokemon.attack()
heal_self = 0
print("You dealt",damage_to_mew,"damage.")
if mew_choice == 1 or mew_choice ==2:
damage_to_user = mew.attack()
heal_mew = 0
print("Mew dealt", damage_to_user, "damage.")
if attack_choice == 3:
heal_self = user_pokemon.heal()
damage_to_mew = 0
print("You healed",heal_self,"health points.")
if mew_choice == 3:
heal_mew = mew.heal()
damage_to_user = 0
print("Mew healed", heal_mew, "health points.")
user_health = user_health - damage_to_user + heal_self
mew_health = mew_health - damage_to_mew + heal_mew
# Pokemon health points are limited by a min of 0 and a max of 100.
if user_health > 100:
user_health = 100
elif user_health <= 0:
user_health = 0
battle_continue = False
if mew_health > 100:
mew_health = 100
elif mew_health <= 0:
mew_health = 0
battle_continue = False
print("Your current health is", user_health)
print("Mew's current health is", mew_health)
print("Your final health is", user_health)
print("Mew's final health is", mew_health)
if user_health < mew_health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
python beginner python-3.x battle-simulation pokemon
New contributor
$endgroup$
add a comment |
$begingroup$
This is based off the popular Pokemon Turn Based project (here).
GOAL
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
SUBGOALS
When someone is defeated, make sure the game prints out that their health has reached 0, and not a negative number.
When the computer's health reaches a set amount (such as 35%), increase it's chance to cast heal.
Give each move a name.
I am new to Python and I am trying to use OOP. Did I use my classes properly and is there a way to reduce the number of if statements in my code? Since this is my first code review, how is the formatting, number of comments, etc?
import random
class Pokemon:
"""Blueprint for turn based Pokemon battle"""
def __init__(self, attack_choice):
self.__attack_choice = attack_choice
def attack(self):
if self.__attack_choice == 1:
attack_points = random.randint(18,25)
return attack_points
elif self.__attack_choice == 2:
attack_points = random.randint(10,35)
return attack_points
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18,35)
return heal_points
###########################################################################
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = eval(input("nSelect an attack: "))
# Mew selects an attack, but focuses on attacking if health is full.
if mew_health == 100:
mew_choice = random.randint(1,2)
else:
mew_choice = random.randint(1,3)
mew = Pokemon(mew_choice)
user_pokemon = Pokemon(attack_choice)
# Attacks by user and Mew are done simultaneously.
if attack_choice == 1 or attack_choice == 2:
damage_to_mew = user_pokemon.attack()
heal_self = 0
print("You dealt",damage_to_mew,"damage.")
if mew_choice == 1 or mew_choice ==2:
damage_to_user = mew.attack()
heal_mew = 0
print("Mew dealt", damage_to_user, "damage.")
if attack_choice == 3:
heal_self = user_pokemon.heal()
damage_to_mew = 0
print("You healed",heal_self,"health points.")
if mew_choice == 3:
heal_mew = mew.heal()
damage_to_user = 0
print("Mew healed", heal_mew, "health points.")
user_health = user_health - damage_to_user + heal_self
mew_health = mew_health - damage_to_mew + heal_mew
# Pokemon health points are limited by a min of 0 and a max of 100.
if user_health > 100:
user_health = 100
elif user_health <= 0:
user_health = 0
battle_continue = False
if mew_health > 100:
mew_health = 100
elif mew_health <= 0:
mew_health = 0
battle_continue = False
print("Your current health is", user_health)
print("Mew's current health is", mew_health)
print("Your final health is", user_health)
print("Mew's final health is", mew_health)
if user_health < mew_health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
python beginner python-3.x battle-simulation pokemon
New contributor
$endgroup$
1
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25
$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago
add a comment |
$begingroup$
This is based off the popular Pokemon Turn Based project (here).
GOAL
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
SUBGOALS
When someone is defeated, make sure the game prints out that their health has reached 0, and not a negative number.
When the computer's health reaches a set amount (such as 35%), increase it's chance to cast heal.
Give each move a name.
I am new to Python and I am trying to use OOP. Did I use my classes properly and is there a way to reduce the number of if statements in my code? Since this is my first code review, how is the formatting, number of comments, etc?
import random
class Pokemon:
"""Blueprint for turn based Pokemon battle"""
def __init__(self, attack_choice):
self.__attack_choice = attack_choice
def attack(self):
if self.__attack_choice == 1:
attack_points = random.randint(18,25)
return attack_points
elif self.__attack_choice == 2:
attack_points = random.randint(10,35)
return attack_points
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18,35)
return heal_points
###########################################################################
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = eval(input("nSelect an attack: "))
# Mew selects an attack, but focuses on attacking if health is full.
if mew_health == 100:
mew_choice = random.randint(1,2)
else:
mew_choice = random.randint(1,3)
mew = Pokemon(mew_choice)
user_pokemon = Pokemon(attack_choice)
# Attacks by user and Mew are done simultaneously.
if attack_choice == 1 or attack_choice == 2:
damage_to_mew = user_pokemon.attack()
heal_self = 0
print("You dealt",damage_to_mew,"damage.")
if mew_choice == 1 or mew_choice ==2:
damage_to_user = mew.attack()
heal_mew = 0
print("Mew dealt", damage_to_user, "damage.")
if attack_choice == 3:
heal_self = user_pokemon.heal()
damage_to_mew = 0
print("You healed",heal_self,"health points.")
if mew_choice == 3:
heal_mew = mew.heal()
damage_to_user = 0
print("Mew healed", heal_mew, "health points.")
user_health = user_health - damage_to_user + heal_self
mew_health = mew_health - damage_to_mew + heal_mew
# Pokemon health points are limited by a min of 0 and a max of 100.
if user_health > 100:
user_health = 100
elif user_health <= 0:
user_health = 0
battle_continue = False
if mew_health > 100:
mew_health = 100
elif mew_health <= 0:
mew_health = 0
battle_continue = False
print("Your current health is", user_health)
print("Mew's current health is", mew_health)
print("Your final health is", user_health)
print("Mew's final health is", mew_health)
if user_health < mew_health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
python beginner python-3.x battle-simulation pokemon
New contributor
$endgroup$
This is based off the popular Pokemon Turn Based project (here).
GOAL
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
SUBGOALS
When someone is defeated, make sure the game prints out that their health has reached 0, and not a negative number.
When the computer's health reaches a set amount (such as 35%), increase it's chance to cast heal.
Give each move a name.
I am new to Python and I am trying to use OOP. Did I use my classes properly and is there a way to reduce the number of if statements in my code? Since this is my first code review, how is the formatting, number of comments, etc?
import random
class Pokemon:
"""Blueprint for turn based Pokemon battle"""
def __init__(self, attack_choice):
self.__attack_choice = attack_choice
def attack(self):
if self.__attack_choice == 1:
attack_points = random.randint(18,25)
return attack_points
elif self.__attack_choice == 2:
attack_points = random.randint(10,35)
return attack_points
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18,35)
return heal_points
###########################################################################
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = eval(input("nSelect an attack: "))
# Mew selects an attack, but focuses on attacking if health is full.
if mew_health == 100:
mew_choice = random.randint(1,2)
else:
mew_choice = random.randint(1,3)
mew = Pokemon(mew_choice)
user_pokemon = Pokemon(attack_choice)
# Attacks by user and Mew are done simultaneously.
if attack_choice == 1 or attack_choice == 2:
damage_to_mew = user_pokemon.attack()
heal_self = 0
print("You dealt",damage_to_mew,"damage.")
if mew_choice == 1 or mew_choice ==2:
damage_to_user = mew.attack()
heal_mew = 0
print("Mew dealt", damage_to_user, "damage.")
if attack_choice == 3:
heal_self = user_pokemon.heal()
damage_to_mew = 0
print("You healed",heal_self,"health points.")
if mew_choice == 3:
heal_mew = mew.heal()
damage_to_user = 0
print("Mew healed", heal_mew, "health points.")
user_health = user_health - damage_to_user + heal_self
mew_health = mew_health - damage_to_mew + heal_mew
# Pokemon health points are limited by a min of 0 and a max of 100.
if user_health > 100:
user_health = 100
elif user_health <= 0:
user_health = 0
battle_continue = False
if mew_health > 100:
mew_health = 100
elif mew_health <= 0:
mew_health = 0
battle_continue = False
print("Your current health is", user_health)
print("Mew's current health is", mew_health)
print("Your final health is", user_health)
print("Mew's final health is", mew_health)
if user_health < mew_health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
python beginner python-3.x battle-simulation pokemon
python beginner python-3.x battle-simulation pokemon
New contributor
New contributor
edited 5 hours ago
200_success
131k17157422
131k17157422
New contributor
asked 7 hours ago
yummylipbalmyummylipbalm
161
161
New contributor
New contributor
1
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25
$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago
add a comment |
1
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25
$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago
1
1
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:
heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:
heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago
add a comment |
4 Answers
4
active
oldest
votes
$begingroup$
The heal
method and the handling of health points strikes me as odd. In the current setup, heal
does not heal; it simply returns a random number. It also currently has no direct relation to any part of the Pokemon class, so it doesn't really make sense as a method of the class. The health points of each Pokemon are "loose" in the script, along with instances of the Pokemon class. You have a Pokemon class representing a Pokemon, and it makes sense that health would be an attribute of a Pokemon itself. Something like (stripped down):
class Pokemon:
def __init__(self, start_health):
self.hp = start_health
def heal(self, heal_amount):
self.hp += heal_amount
def hurt(self, damage):
self.hp -= damage
# Or, to reuse existing methods
#self.heal(-damage)
Now, you can do something like:
mew = Pokemon(100)
mew.hurt(50) # Ow
mew.heal(49) # Back to almost full
print(mew.hp) # Prints 99
And you don't need to have loose health values floating around in the code for each Pokemon in use. Using the method like this also lets you check the health after healing to ensure that you didn't exceed the maximum health allowed. As long as you're going to make use of a class, you can benefit by having it encapsulate all the data (like health) that's directly relevant to it.
I also decided to not have heal
generate the random amount to heal. That can cause problems with testing since you can't force it to heal by a certain amount. Allow the user to pick how the data is supplied unless you have a good reason to restrict it. If they want to hurt the Pokemon by a random amount, they can generate the random data themselves to pass it in.
while battle_continue == True:
has a redundant condition. while
(and if
) just check if their condition is truthy. You can just write:
while battle_continue:
$endgroup$
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
add a comment |
$begingroup$
Hello and welcome to CodeReview. Congratulations on writing code that is well-formatted and compliant with the basics of the generally-agreed-upon Python coding style.
You have made some errors in organization and structure, however. Let's look at those first:
Organization & Structure
Organization
Your code is "against the wall." This means that if I import your module, the code is going to be run, not just loaded. That's bad because it means you can't import it into the python command line REPL to play around with it, and you can load it into a debugger or a test framework.
The basic solution is to find all your "left-edge" statements, like these:
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
And move them into a function. Call it main
until you have a better name (if you have a lot of code you might break it into more than one function, but main
is a good place to start!). Then at the bottom of your module, do:
if __name__ == '__main__:
main()
That "hook" means that if you run python myfile.py
the code will run and the game will play, but if you start the REPL using just python
and then type >>> from myfile import *
you will have all the functions available and can call them to see what they do in different circumstances. And it also means you can use one of the many Python unit testing frameworks to check your code.
Structure
You have one class, and you don't really use it. The class is syntactically correct, but you aren't treating it as a class because you don't "trust" it. You're still treating it as a collection of functions that you can call.
Let's look at the first part of the problem statement. It's nice and small, so let's just play a game of find the nouns and see if that gives us any ideas about objects:
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
(I flagged "heal" as a noun because it's really the opposite of "damage".)
The general rule for OOP is that objects are nouns and methods are verbs. So our set of potential objects includes:
- Game
- User
- Computer
- Turn
- Move
- Health
- Damage
- Range
- Heal
- Message
With those in mind, it seems like User and Computer are likely to be either different parts (methods or functions) of the Game object, or they will be two different implementations of a Player interface (or subclasses of a Player class).
The Turn might be an object, but it seems more likely that "take turns" is a verb on the Game. A Move seems very likely to be an object, since there are so many details and requirements about them.
Health seems like an attribute, since we get a starting number provided, and both Heal and Damage both seem like verbs affecting health more than separate attributes or objects. The Range seems like an attribute of the various Move objects, but an internal attribute - when you invoke a move, it will adjust a player's health by some internally-computed amount.
Message is probably just a string. It's an object, but of a type that comes built-in. There is one key question, though: are turns sequential or simultaneous? If turns are sequential, messages can just be printed as they happen. If turns are simultaneous, messages might have to be computed after the results of both moves are understood. (What happens if both players kill each other during the same turn?)
Pro-Tips
Move class
Having Move
as a class would be an example of the Command pattern, and might look something like this:
from abc import ABC, abstractmethod
class Move(ABC):
''' Abstract game move. (Command pattern.) Perform an action when
invoked.
'''
def __init__(self, name):
self.name = name
@abstractmethod
def execute(self, us: 'Player', them: 'Player') -> str:
''' Perform an operation like damaging them, or healing us. '''
return f"us.name uses self.name! It's super effective!"
(Note: You could also construct the Move objects with the us/them attributes configured during __init__
, instead of passed as parameters.)
Random choices
You should eliminate all uses of if
that involve random choices. You are randomly choosing a move: use a list or tuple of Move objects and compute the index of the object, then pass that object around. You are randomly generating damage or healing: compute the number, then apply that number to the appropriate player health attribute. The only if
statement you need is to guard against going above or below the maximum or minimum health values (and those aren't random)!
$endgroup$
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
add a comment |
$begingroup$
Conceptual
You have chosen a great problem to begin with, however there are a few things you get wrong about OOP. OOP is not simply about using a class for "everything", it's a kind of way to think about a problem.
To begin with, it helped me a lot to think about a class as a prototype of a "thing". In your case the "thing" would be a Pokemon. A Pokemon can do certain things. In your simplified versions that would be 1. attack another Pokemon and 2. heal itself. Often these actions are reflected in the classes's methods. I think you mostly understood that. What else is there about the Pokemon/"thing" it has certain properties that describe it. I would say that's an aspect you did not think about. A property could be name, color, ... or it's health status. We also learn that the health can only be between 0
and 100
.
So with this on our mind, let's think of a new design for class Pokemon
:
class Pokemon:
"""Blueprint for a new pokemon"""
def __init__(self):
self._health = 100
# ^--- the leading _ is a convention to mark internal values
@property
def health(self):
"""The health of the Pokemon which is between 0 and 100"""
return self._health
@health.setter
def health(self, new_health):
"""Set the new heath value"""
# here the health limits are enforced
self._health = min(100, max(0, new_health))
def attack(self, other, choice):
"""Attack another Pokemon with the chosen attack (1 or 2)
This function also returns the raw amount of random damage dealt. The
amount of damage dealt depends on the attack type.
"""
if choice == 1:
attack_points = random.randint(18, 25)
elif choice == 2:
attack_points = random.randint(10, 35)
else:
print("That is not a selection. You lost your turn!")
attack_points = 0
other.health -= attack_points
return attack_points
def heal(self):
"""Heal the Pokemon"""
heal_points = random.randint(18, 35)
self.health += heal_points
return heal_points
A lot of this should look familiar to you, since the main work is still done by the code you've written. What is new that the Pokemon now has health. If your unfamiliar with properties in Python just think of them as synthatic sugar on what would normally be done using getter and setter functions in other languages. There is a great SO post with a nice explanation on properties. In addition to that, attack
and heal
now handle the update of the health value for the Pokemon, as well as the opponent. This allows us to write up a simple battle in a very concise way:
mew = Pokemon()
user = Pokemon()
mew.attack(user, 1)
print(f"User health after attack: user.health")
user.heal()
print(f"User health after heal: user.health")
mew.heal() # health should still only be 100
print(f"Mew's health after 'illegal' heal: user.health")
which prints for example:
User health after attack: 75
User health after heal: 100
Mew's health after 'illegal' heal: 100
No additional variables that need to track health status, no checking on the health limits. Everything is nicely encapsulated into the Pokemon
class.
Short break on style and other conventions
Another thing that has changed in contrast to your solution is the documentation I added to each function. Python has an "official" Style Guide (which is worth a read on its own) with a section on how to write these so called docstrings here.
It also features guidelines on where to use blank lines and where not to use them. In my oppinion the excessive use of blank lines in your code hinders readability more than it does help to structure the code.
I also used only a single leading underscore for my internal value instead of two as you did. The Style Guide also has you covered on this topic. In general you should always use a single leading underscore to mark functions and variables/members as internal. For more details on what happens if you use two leading underscores, follow the link above.
After that short intermezzo, let's look on how the battle simulation looks with the new class:
def battle_simulation():
"""Run a simple interactive Pokemon battle simulation"""
mew = Pokemon()
user_pokemon = Pokemon()
while True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = int(input("nSelect an attack: "))
# DON'T use eval on user input, this can be dangerous!
# Mew selects an attack, but focuses on attacking if health is full.
mew_choice = random.randint(1, 2 if mew.health == 100 else 3)
# this is your original distinction just condensed into a single line
# Attacks by user and Mew are done simultaneously
# with the changes to Pokemon, there is no need to save all the
# intermediate damage/heal values -> nice and short code
if attack_choice != 3:
print(f"You dealt user_pokemon.attack(mew, attack_choice) damage.")
if mew_choice != 3:
print(f"Mew dealt mew.attack(user_pokemon, mew_choice) damage.")
if attack_choice == 3:
print(f"You healed user_pokemon.heal() health points.")
if mew_choice == 3:
print(f"Mew healed mew.heal() health points.")
if mew.health == 0 or user_pokemon.health == 0:
break
print(f"Your current health is user_pokemon.health")
print(f"Mew's current health is mew.health")
print(f"Your final health is user_pokemon.health")
print(f"Mew's final health is mew.health")
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
if __name__ == "__main__":
battle_simulation()
As you can see, we got rid of all the variables needed to store damage and heal values since everything is no done internally in the Pokemon
class. With this, the code becomes quite a bit shorter, clearer and easier to read. Cool, isn't it?
A thing you will see me use quite often are the so called f-strings. They take arbitrary Python expressions (function calls, variables, ...) and allow to incorporate them into a formatted string. If you want to know more about them, I would recommend this blog post.
Happy Coding!
$endgroup$
add a comment |
$begingroup$
1) The doc comment in the class should give information for that class and not for the whole program.
2) I don't see why the Pokemon should be initialised with an attack choice if it can do multiple attacks. I think the choice should be passed as a parameter to the attack method. Instead it will make more sense for the class to hold the health and the logic for the damage which is currently in the while
loop.
3) battle_continue == True
can be simplified to only battle_continue
and can be renamed to battle_in_progress
.
4) The print statement: print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
can be separated into multiple lines for better readability.
5) The moves can be an Enum
instead of magic numbers.
6) The move selection should have exception handling for invalid inputs.
7) The attack function can take as a parameter the target of the attack and deduce the life points.
8) Some of the other functionality is better suited as methods on the class e.g checking if the health is 0, applying damage to a Pokemon, healing.
Final Implementation:
import random
from enum import Enum
class Pokemon:
"""Class for representing a pokemon"""
def __init__(self, name):
self.name = name
self.health = 100
def make_move(self, move, target):
if move == Moves.HEAL:
self.heal()
else:
self.attack(move, target)
def attack(self, move, target):
attack_points = None
if move == Moves.CLOSE_ATTACK:
attack_points = random.randint(18, 25)
if move == Moves.RANGE_ATTACK:
attack_points = random.randint(10, 35)
if attack_points:
target.take_damage(attack_points)
print(self.name, "dealt", attack_points, "damage.")
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18, 35)
self.health = self.health + heal_points
print(self.name, "healed", heal_points, "health points.")
if self.health > 100:
self.health = 100
def take_damage(self, points):
self.health = self.health - points
if self.health < 0:
self.health = 0
def is_dead(self):
return self.health <= 0
class PokemonBot(Pokemon):
def select_move(self):
if self.health == 100:
choice = Moves.get_random_attack()
else:
choice = Moves.get_random_move()
return choice
class Moves(Enum):
CLOSE_ATTACK = 1
RANGE_ATTACK = 2
HEAL = 3
@staticmethod
def get_random_attack():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK])
@staticmethod
def get_random_move():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK, Moves.HEAL])
###########################################################################
battle_in_progress = True
mew = PokemonBot("Mew")
name = input("Name your pokemon: ")
user_pokemon = Pokemon(name)
while battle_in_progress:
print("MOVE CHOICES")
print("1. Close range attack")
print("2. Far range attack")
print("3. Heal")
move_choice = None
while move_choice is None:
try:
move_choice = Moves(int(input("Select a move: ")))
except ValueError:
print("Please enter a valid move!")
mew_move = mew.select_move()
user_pokemon.make_move(move_choice, mew)
mew.make_move(mew_move, user_pokemon)
print("Current health of", user_pokemon.name, "is", user_pokemon.health)
print("Current health of", mew.name, "is", mew.health)
if mew.is_dead() or user_pokemon.is_dead():
battle_in_progress = False
print("Final health of", user_pokemon.name, "is", user_pokemon.health)
print("Final health of", mew.name, "is", mew.health)
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against", mew.name, "!")
$endgroup$
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
add a comment |
Your Answer
StackExchange.ifUsing("editor", function ()
return StackExchange.using("mathjaxEditing", function ()
StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix)
StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
);
);
, "mathjax-editing");
StackExchange.ifUsing("editor", function ()
StackExchange.using("externalEditor", function ()
StackExchange.using("snippets", function ()
StackExchange.snippets.init();
);
);
, "code-snippets");
StackExchange.ready(function()
var channelOptions =
tags: "".split(" "),
id: "196"
;
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function()
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled)
StackExchange.using("snippets", function()
createEditor();
);
else
createEditor();
);
function createEditor()
StackExchange.prepareEditor(
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader:
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
,
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
);
);
yummylipbalm is a new contributor. Be nice, and check out our Code of Conduct.
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f217222%2fpokemon-turn-based-battle-python%23new-answer', 'question_page');
);
Post as a guest
Required, but never shown
4 Answers
4
active
oldest
votes
4 Answers
4
active
oldest
votes
active
oldest
votes
active
oldest
votes
$begingroup$
The heal
method and the handling of health points strikes me as odd. In the current setup, heal
does not heal; it simply returns a random number. It also currently has no direct relation to any part of the Pokemon class, so it doesn't really make sense as a method of the class. The health points of each Pokemon are "loose" in the script, along with instances of the Pokemon class. You have a Pokemon class representing a Pokemon, and it makes sense that health would be an attribute of a Pokemon itself. Something like (stripped down):
class Pokemon:
def __init__(self, start_health):
self.hp = start_health
def heal(self, heal_amount):
self.hp += heal_amount
def hurt(self, damage):
self.hp -= damage
# Or, to reuse existing methods
#self.heal(-damage)
Now, you can do something like:
mew = Pokemon(100)
mew.hurt(50) # Ow
mew.heal(49) # Back to almost full
print(mew.hp) # Prints 99
And you don't need to have loose health values floating around in the code for each Pokemon in use. Using the method like this also lets you check the health after healing to ensure that you didn't exceed the maximum health allowed. As long as you're going to make use of a class, you can benefit by having it encapsulate all the data (like health) that's directly relevant to it.
I also decided to not have heal
generate the random amount to heal. That can cause problems with testing since you can't force it to heal by a certain amount. Allow the user to pick how the data is supplied unless you have a good reason to restrict it. If they want to hurt the Pokemon by a random amount, they can generate the random data themselves to pass it in.
while battle_continue == True:
has a redundant condition. while
(and if
) just check if their condition is truthy. You can just write:
while battle_continue:
$endgroup$
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
add a comment |
$begingroup$
The heal
method and the handling of health points strikes me as odd. In the current setup, heal
does not heal; it simply returns a random number. It also currently has no direct relation to any part of the Pokemon class, so it doesn't really make sense as a method of the class. The health points of each Pokemon are "loose" in the script, along with instances of the Pokemon class. You have a Pokemon class representing a Pokemon, and it makes sense that health would be an attribute of a Pokemon itself. Something like (stripped down):
class Pokemon:
def __init__(self, start_health):
self.hp = start_health
def heal(self, heal_amount):
self.hp += heal_amount
def hurt(self, damage):
self.hp -= damage
# Or, to reuse existing methods
#self.heal(-damage)
Now, you can do something like:
mew = Pokemon(100)
mew.hurt(50) # Ow
mew.heal(49) # Back to almost full
print(mew.hp) # Prints 99
And you don't need to have loose health values floating around in the code for each Pokemon in use. Using the method like this also lets you check the health after healing to ensure that you didn't exceed the maximum health allowed. As long as you're going to make use of a class, you can benefit by having it encapsulate all the data (like health) that's directly relevant to it.
I also decided to not have heal
generate the random amount to heal. That can cause problems with testing since you can't force it to heal by a certain amount. Allow the user to pick how the data is supplied unless you have a good reason to restrict it. If they want to hurt the Pokemon by a random amount, they can generate the random data themselves to pass it in.
while battle_continue == True:
has a redundant condition. while
(and if
) just check if their condition is truthy. You can just write:
while battle_continue:
$endgroup$
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
add a comment |
$begingroup$
The heal
method and the handling of health points strikes me as odd. In the current setup, heal
does not heal; it simply returns a random number. It also currently has no direct relation to any part of the Pokemon class, so it doesn't really make sense as a method of the class. The health points of each Pokemon are "loose" in the script, along with instances of the Pokemon class. You have a Pokemon class representing a Pokemon, and it makes sense that health would be an attribute of a Pokemon itself. Something like (stripped down):
class Pokemon:
def __init__(self, start_health):
self.hp = start_health
def heal(self, heal_amount):
self.hp += heal_amount
def hurt(self, damage):
self.hp -= damage
# Or, to reuse existing methods
#self.heal(-damage)
Now, you can do something like:
mew = Pokemon(100)
mew.hurt(50) # Ow
mew.heal(49) # Back to almost full
print(mew.hp) # Prints 99
And you don't need to have loose health values floating around in the code for each Pokemon in use. Using the method like this also lets you check the health after healing to ensure that you didn't exceed the maximum health allowed. As long as you're going to make use of a class, you can benefit by having it encapsulate all the data (like health) that's directly relevant to it.
I also decided to not have heal
generate the random amount to heal. That can cause problems with testing since you can't force it to heal by a certain amount. Allow the user to pick how the data is supplied unless you have a good reason to restrict it. If they want to hurt the Pokemon by a random amount, they can generate the random data themselves to pass it in.
while battle_continue == True:
has a redundant condition. while
(and if
) just check if their condition is truthy. You can just write:
while battle_continue:
$endgroup$
The heal
method and the handling of health points strikes me as odd. In the current setup, heal
does not heal; it simply returns a random number. It also currently has no direct relation to any part of the Pokemon class, so it doesn't really make sense as a method of the class. The health points of each Pokemon are "loose" in the script, along with instances of the Pokemon class. You have a Pokemon class representing a Pokemon, and it makes sense that health would be an attribute of a Pokemon itself. Something like (stripped down):
class Pokemon:
def __init__(self, start_health):
self.hp = start_health
def heal(self, heal_amount):
self.hp += heal_amount
def hurt(self, damage):
self.hp -= damage
# Or, to reuse existing methods
#self.heal(-damage)
Now, you can do something like:
mew = Pokemon(100)
mew.hurt(50) # Ow
mew.heal(49) # Back to almost full
print(mew.hp) # Prints 99
And you don't need to have loose health values floating around in the code for each Pokemon in use. Using the method like this also lets you check the health after healing to ensure that you didn't exceed the maximum health allowed. As long as you're going to make use of a class, you can benefit by having it encapsulate all the data (like health) that's directly relevant to it.
I also decided to not have heal
generate the random amount to heal. That can cause problems with testing since you can't force it to heal by a certain amount. Allow the user to pick how the data is supplied unless you have a good reason to restrict it. If they want to hurt the Pokemon by a random amount, they can generate the random data themselves to pass it in.
while battle_continue == True:
has a redundant condition. while
(and if
) just check if their condition is truthy. You can just write:
while battle_continue:
edited 4 hours ago
answered 5 hours ago
CarcigenicateCarcigenicate
3,98911632
3,98911632
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
add a comment |
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
$begingroup$
The wording of all of this is very off, but I'm quite tired and can't seem to make it any better. Hopefully it's still clear.
$endgroup$
– Carcigenicate
5 hours ago
add a comment |
$begingroup$
Hello and welcome to CodeReview. Congratulations on writing code that is well-formatted and compliant with the basics of the generally-agreed-upon Python coding style.
You have made some errors in organization and structure, however. Let's look at those first:
Organization & Structure
Organization
Your code is "against the wall." This means that if I import your module, the code is going to be run, not just loaded. That's bad because it means you can't import it into the python command line REPL to play around with it, and you can load it into a debugger or a test framework.
The basic solution is to find all your "left-edge" statements, like these:
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
And move them into a function. Call it main
until you have a better name (if you have a lot of code you might break it into more than one function, but main
is a good place to start!). Then at the bottom of your module, do:
if __name__ == '__main__:
main()
That "hook" means that if you run python myfile.py
the code will run and the game will play, but if you start the REPL using just python
and then type >>> from myfile import *
you will have all the functions available and can call them to see what they do in different circumstances. And it also means you can use one of the many Python unit testing frameworks to check your code.
Structure
You have one class, and you don't really use it. The class is syntactically correct, but you aren't treating it as a class because you don't "trust" it. You're still treating it as a collection of functions that you can call.
Let's look at the first part of the problem statement. It's nice and small, so let's just play a game of find the nouns and see if that gives us any ideas about objects:
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
(I flagged "heal" as a noun because it's really the opposite of "damage".)
The general rule for OOP is that objects are nouns and methods are verbs. So our set of potential objects includes:
- Game
- User
- Computer
- Turn
- Move
- Health
- Damage
- Range
- Heal
- Message
With those in mind, it seems like User and Computer are likely to be either different parts (methods or functions) of the Game object, or they will be two different implementations of a Player interface (or subclasses of a Player class).
The Turn might be an object, but it seems more likely that "take turns" is a verb on the Game. A Move seems very likely to be an object, since there are so many details and requirements about them.
Health seems like an attribute, since we get a starting number provided, and both Heal and Damage both seem like verbs affecting health more than separate attributes or objects. The Range seems like an attribute of the various Move objects, but an internal attribute - when you invoke a move, it will adjust a player's health by some internally-computed amount.
Message is probably just a string. It's an object, but of a type that comes built-in. There is one key question, though: are turns sequential or simultaneous? If turns are sequential, messages can just be printed as they happen. If turns are simultaneous, messages might have to be computed after the results of both moves are understood. (What happens if both players kill each other during the same turn?)
Pro-Tips
Move class
Having Move
as a class would be an example of the Command pattern, and might look something like this:
from abc import ABC, abstractmethod
class Move(ABC):
''' Abstract game move. (Command pattern.) Perform an action when
invoked.
'''
def __init__(self, name):
self.name = name
@abstractmethod
def execute(self, us: 'Player', them: 'Player') -> str:
''' Perform an operation like damaging them, or healing us. '''
return f"us.name uses self.name! It's super effective!"
(Note: You could also construct the Move objects with the us/them attributes configured during __init__
, instead of passed as parameters.)
Random choices
You should eliminate all uses of if
that involve random choices. You are randomly choosing a move: use a list or tuple of Move objects and compute the index of the object, then pass that object around. You are randomly generating damage or healing: compute the number, then apply that number to the appropriate player health attribute. The only if
statement you need is to guard against going above or below the maximum or minimum health values (and those aren't random)!
$endgroup$
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
add a comment |
$begingroup$
Hello and welcome to CodeReview. Congratulations on writing code that is well-formatted and compliant with the basics of the generally-agreed-upon Python coding style.
You have made some errors in organization and structure, however. Let's look at those first:
Organization & Structure
Organization
Your code is "against the wall." This means that if I import your module, the code is going to be run, not just loaded. That's bad because it means you can't import it into the python command line REPL to play around with it, and you can load it into a debugger or a test framework.
The basic solution is to find all your "left-edge" statements, like these:
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
And move them into a function. Call it main
until you have a better name (if you have a lot of code you might break it into more than one function, but main
is a good place to start!). Then at the bottom of your module, do:
if __name__ == '__main__:
main()
That "hook" means that if you run python myfile.py
the code will run and the game will play, but if you start the REPL using just python
and then type >>> from myfile import *
you will have all the functions available and can call them to see what they do in different circumstances. And it also means you can use one of the many Python unit testing frameworks to check your code.
Structure
You have one class, and you don't really use it. The class is syntactically correct, but you aren't treating it as a class because you don't "trust" it. You're still treating it as a collection of functions that you can call.
Let's look at the first part of the problem statement. It's nice and small, so let's just play a game of find the nouns and see if that gives us any ideas about objects:
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
(I flagged "heal" as a noun because it's really the opposite of "damage".)
The general rule for OOP is that objects are nouns and methods are verbs. So our set of potential objects includes:
- Game
- User
- Computer
- Turn
- Move
- Health
- Damage
- Range
- Heal
- Message
With those in mind, it seems like User and Computer are likely to be either different parts (methods or functions) of the Game object, or they will be two different implementations of a Player interface (or subclasses of a Player class).
The Turn might be an object, but it seems more likely that "take turns" is a verb on the Game. A Move seems very likely to be an object, since there are so many details and requirements about them.
Health seems like an attribute, since we get a starting number provided, and both Heal and Damage both seem like verbs affecting health more than separate attributes or objects. The Range seems like an attribute of the various Move objects, but an internal attribute - when you invoke a move, it will adjust a player's health by some internally-computed amount.
Message is probably just a string. It's an object, but of a type that comes built-in. There is one key question, though: are turns sequential or simultaneous? If turns are sequential, messages can just be printed as they happen. If turns are simultaneous, messages might have to be computed after the results of both moves are understood. (What happens if both players kill each other during the same turn?)
Pro-Tips
Move class
Having Move
as a class would be an example of the Command pattern, and might look something like this:
from abc import ABC, abstractmethod
class Move(ABC):
''' Abstract game move. (Command pattern.) Perform an action when
invoked.
'''
def __init__(self, name):
self.name = name
@abstractmethod
def execute(self, us: 'Player', them: 'Player') -> str:
''' Perform an operation like damaging them, or healing us. '''
return f"us.name uses self.name! It's super effective!"
(Note: You could also construct the Move objects with the us/them attributes configured during __init__
, instead of passed as parameters.)
Random choices
You should eliminate all uses of if
that involve random choices. You are randomly choosing a move: use a list or tuple of Move objects and compute the index of the object, then pass that object around. You are randomly generating damage or healing: compute the number, then apply that number to the appropriate player health attribute. The only if
statement you need is to guard against going above or below the maximum or minimum health values (and those aren't random)!
$endgroup$
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
add a comment |
$begingroup$
Hello and welcome to CodeReview. Congratulations on writing code that is well-formatted and compliant with the basics of the generally-agreed-upon Python coding style.
You have made some errors in organization and structure, however. Let's look at those first:
Organization & Structure
Organization
Your code is "against the wall." This means that if I import your module, the code is going to be run, not just loaded. That's bad because it means you can't import it into the python command line REPL to play around with it, and you can load it into a debugger or a test framework.
The basic solution is to find all your "left-edge" statements, like these:
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
And move them into a function. Call it main
until you have a better name (if you have a lot of code you might break it into more than one function, but main
is a good place to start!). Then at the bottom of your module, do:
if __name__ == '__main__:
main()
That "hook" means that if you run python myfile.py
the code will run and the game will play, but if you start the REPL using just python
and then type >>> from myfile import *
you will have all the functions available and can call them to see what they do in different circumstances. And it also means you can use one of the many Python unit testing frameworks to check your code.
Structure
You have one class, and you don't really use it. The class is syntactically correct, but you aren't treating it as a class because you don't "trust" it. You're still treating it as a collection of functions that you can call.
Let's look at the first part of the problem statement. It's nice and small, so let's just play a game of find the nouns and see if that gives us any ideas about objects:
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
(I flagged "heal" as a noun because it's really the opposite of "damage".)
The general rule for OOP is that objects are nouns and methods are verbs. So our set of potential objects includes:
- Game
- User
- Computer
- Turn
- Move
- Health
- Damage
- Range
- Heal
- Message
With those in mind, it seems like User and Computer are likely to be either different parts (methods or functions) of the Game object, or they will be two different implementations of a Player interface (or subclasses of a Player class).
The Turn might be an object, but it seems more likely that "take turns" is a verb on the Game. A Move seems very likely to be an object, since there are so many details and requirements about them.
Health seems like an attribute, since we get a starting number provided, and both Heal and Damage both seem like verbs affecting health more than separate attributes or objects. The Range seems like an attribute of the various Move objects, but an internal attribute - when you invoke a move, it will adjust a player's health by some internally-computed amount.
Message is probably just a string. It's an object, but of a type that comes built-in. There is one key question, though: are turns sequential or simultaneous? If turns are sequential, messages can just be printed as they happen. If turns are simultaneous, messages might have to be computed after the results of both moves are understood. (What happens if both players kill each other during the same turn?)
Pro-Tips
Move class
Having Move
as a class would be an example of the Command pattern, and might look something like this:
from abc import ABC, abstractmethod
class Move(ABC):
''' Abstract game move. (Command pattern.) Perform an action when
invoked.
'''
def __init__(self, name):
self.name = name
@abstractmethod
def execute(self, us: 'Player', them: 'Player') -> str:
''' Perform an operation like damaging them, or healing us. '''
return f"us.name uses self.name! It's super effective!"
(Note: You could also construct the Move objects with the us/them attributes configured during __init__
, instead of passed as parameters.)
Random choices
You should eliminate all uses of if
that involve random choices. You are randomly choosing a move: use a list or tuple of Move objects and compute the index of the object, then pass that object around. You are randomly generating damage or healing: compute the number, then apply that number to the appropriate player health attribute. The only if
statement you need is to guard against going above or below the maximum or minimum health values (and those aren't random)!
$endgroup$
Hello and welcome to CodeReview. Congratulations on writing code that is well-formatted and compliant with the basics of the generally-agreed-upon Python coding style.
You have made some errors in organization and structure, however. Let's look at those first:
Organization & Structure
Organization
Your code is "against the wall." This means that if I import your module, the code is going to be run, not just loaded. That's bad because it means you can't import it into the python command line REPL to play around with it, and you can load it into a debugger or a test framework.
The basic solution is to find all your "left-edge" statements, like these:
user_health = 100
mew_health = 100
battle_continue = True
while battle_continue == True:
And move them into a function. Call it main
until you have a better name (if you have a lot of code you might break it into more than one function, but main
is a good place to start!). Then at the bottom of your module, do:
if __name__ == '__main__:
main()
That "hook" means that if you run python myfile.py
the code will run and the game will play, but if you start the REPL using just python
and then type >>> from myfile import *
you will have all the functions available and can call them to see what they do in different circumstances. And it also means you can use one of the many Python unit testing frameworks to check your code.
Structure
You have one class, and you don't really use it. The class is syntactically correct, but you aren't treating it as a class because you don't "trust" it. You're still treating it as a collection of functions that you can call.
Let's look at the first part of the problem statement. It's nice and small, so let's just play a game of find the nouns and see if that gives us any ideas about objects:
Write a simple game that allows the user and the computer to take
turns selecting moves to use against each other. Both the computer and
the player should start out at the same amount of health (such as
100), and should be able to choose between the three moves:
The first move should do moderate damage and has a small range (such as 18-25).
The second move should have a large range of damage and can deal high or low damage (such as 10-35).
The third move should heal whoever casts it a moderate amount, similar to the first move.
After each move, a message should be printed out that tells the user
what just happened, and how much health the user and computer have.
Once the user or the computer's health reaches 0, the game should end.
(I flagged "heal" as a noun because it's really the opposite of "damage".)
The general rule for OOP is that objects are nouns and methods are verbs. So our set of potential objects includes:
- Game
- User
- Computer
- Turn
- Move
- Health
- Damage
- Range
- Heal
- Message
With those in mind, it seems like User and Computer are likely to be either different parts (methods or functions) of the Game object, or they will be two different implementations of a Player interface (or subclasses of a Player class).
The Turn might be an object, but it seems more likely that "take turns" is a verb on the Game. A Move seems very likely to be an object, since there are so many details and requirements about them.
Health seems like an attribute, since we get a starting number provided, and both Heal and Damage both seem like verbs affecting health more than separate attributes or objects. The Range seems like an attribute of the various Move objects, but an internal attribute - when you invoke a move, it will adjust a player's health by some internally-computed amount.
Message is probably just a string. It's an object, but of a type that comes built-in. There is one key question, though: are turns sequential or simultaneous? If turns are sequential, messages can just be printed as they happen. If turns are simultaneous, messages might have to be computed after the results of both moves are understood. (What happens if both players kill each other during the same turn?)
Pro-Tips
Move class
Having Move
as a class would be an example of the Command pattern, and might look something like this:
from abc import ABC, abstractmethod
class Move(ABC):
''' Abstract game move. (Command pattern.) Perform an action when
invoked.
'''
def __init__(self, name):
self.name = name
@abstractmethod
def execute(self, us: 'Player', them: 'Player') -> str:
''' Perform an operation like damaging them, or healing us. '''
return f"us.name uses self.name! It's super effective!"
(Note: You could also construct the Move objects with the us/them attributes configured during __init__
, instead of passed as parameters.)
Random choices
You should eliminate all uses of if
that involve random choices. You are randomly choosing a move: use a list or tuple of Move objects and compute the index of the object, then pass that object around. You are randomly generating damage or healing: compute the number, then apply that number to the appropriate player health attribute. The only if
statement you need is to guard against going above or below the maximum or minimum health values (and those aren't random)!
answered 5 hours ago
Austin HastingsAustin Hastings
7,6771235
7,6771235
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
add a comment |
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
Interesting idea. Would have never thought of this.
$endgroup$
– Alex
5 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
$begingroup$
TIL "code is against the wall" = a lot of left-adjusted statements / scripting
$endgroup$
– aaaaaa
3 hours ago
add a comment |
$begingroup$
Conceptual
You have chosen a great problem to begin with, however there are a few things you get wrong about OOP. OOP is not simply about using a class for "everything", it's a kind of way to think about a problem.
To begin with, it helped me a lot to think about a class as a prototype of a "thing". In your case the "thing" would be a Pokemon. A Pokemon can do certain things. In your simplified versions that would be 1. attack another Pokemon and 2. heal itself. Often these actions are reflected in the classes's methods. I think you mostly understood that. What else is there about the Pokemon/"thing" it has certain properties that describe it. I would say that's an aspect you did not think about. A property could be name, color, ... or it's health status. We also learn that the health can only be between 0
and 100
.
So with this on our mind, let's think of a new design for class Pokemon
:
class Pokemon:
"""Blueprint for a new pokemon"""
def __init__(self):
self._health = 100
# ^--- the leading _ is a convention to mark internal values
@property
def health(self):
"""The health of the Pokemon which is between 0 and 100"""
return self._health
@health.setter
def health(self, new_health):
"""Set the new heath value"""
# here the health limits are enforced
self._health = min(100, max(0, new_health))
def attack(self, other, choice):
"""Attack another Pokemon with the chosen attack (1 or 2)
This function also returns the raw amount of random damage dealt. The
amount of damage dealt depends on the attack type.
"""
if choice == 1:
attack_points = random.randint(18, 25)
elif choice == 2:
attack_points = random.randint(10, 35)
else:
print("That is not a selection. You lost your turn!")
attack_points = 0
other.health -= attack_points
return attack_points
def heal(self):
"""Heal the Pokemon"""
heal_points = random.randint(18, 35)
self.health += heal_points
return heal_points
A lot of this should look familiar to you, since the main work is still done by the code you've written. What is new that the Pokemon now has health. If your unfamiliar with properties in Python just think of them as synthatic sugar on what would normally be done using getter and setter functions in other languages. There is a great SO post with a nice explanation on properties. In addition to that, attack
and heal
now handle the update of the health value for the Pokemon, as well as the opponent. This allows us to write up a simple battle in a very concise way:
mew = Pokemon()
user = Pokemon()
mew.attack(user, 1)
print(f"User health after attack: user.health")
user.heal()
print(f"User health after heal: user.health")
mew.heal() # health should still only be 100
print(f"Mew's health after 'illegal' heal: user.health")
which prints for example:
User health after attack: 75
User health after heal: 100
Mew's health after 'illegal' heal: 100
No additional variables that need to track health status, no checking on the health limits. Everything is nicely encapsulated into the Pokemon
class.
Short break on style and other conventions
Another thing that has changed in contrast to your solution is the documentation I added to each function. Python has an "official" Style Guide (which is worth a read on its own) with a section on how to write these so called docstrings here.
It also features guidelines on where to use blank lines and where not to use them. In my oppinion the excessive use of blank lines in your code hinders readability more than it does help to structure the code.
I also used only a single leading underscore for my internal value instead of two as you did. The Style Guide also has you covered on this topic. In general you should always use a single leading underscore to mark functions and variables/members as internal. For more details on what happens if you use two leading underscores, follow the link above.
After that short intermezzo, let's look on how the battle simulation looks with the new class:
def battle_simulation():
"""Run a simple interactive Pokemon battle simulation"""
mew = Pokemon()
user_pokemon = Pokemon()
while True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = int(input("nSelect an attack: "))
# DON'T use eval on user input, this can be dangerous!
# Mew selects an attack, but focuses on attacking if health is full.
mew_choice = random.randint(1, 2 if mew.health == 100 else 3)
# this is your original distinction just condensed into a single line
# Attacks by user and Mew are done simultaneously
# with the changes to Pokemon, there is no need to save all the
# intermediate damage/heal values -> nice and short code
if attack_choice != 3:
print(f"You dealt user_pokemon.attack(mew, attack_choice) damage.")
if mew_choice != 3:
print(f"Mew dealt mew.attack(user_pokemon, mew_choice) damage.")
if attack_choice == 3:
print(f"You healed user_pokemon.heal() health points.")
if mew_choice == 3:
print(f"Mew healed mew.heal() health points.")
if mew.health == 0 or user_pokemon.health == 0:
break
print(f"Your current health is user_pokemon.health")
print(f"Mew's current health is mew.health")
print(f"Your final health is user_pokemon.health")
print(f"Mew's final health is mew.health")
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
if __name__ == "__main__":
battle_simulation()
As you can see, we got rid of all the variables needed to store damage and heal values since everything is no done internally in the Pokemon
class. With this, the code becomes quite a bit shorter, clearer and easier to read. Cool, isn't it?
A thing you will see me use quite often are the so called f-strings. They take arbitrary Python expressions (function calls, variables, ...) and allow to incorporate them into a formatted string. If you want to know more about them, I would recommend this blog post.
Happy Coding!
$endgroup$
add a comment |
$begingroup$
Conceptual
You have chosen a great problem to begin with, however there are a few things you get wrong about OOP. OOP is not simply about using a class for "everything", it's a kind of way to think about a problem.
To begin with, it helped me a lot to think about a class as a prototype of a "thing". In your case the "thing" would be a Pokemon. A Pokemon can do certain things. In your simplified versions that would be 1. attack another Pokemon and 2. heal itself. Often these actions are reflected in the classes's methods. I think you mostly understood that. What else is there about the Pokemon/"thing" it has certain properties that describe it. I would say that's an aspect you did not think about. A property could be name, color, ... or it's health status. We also learn that the health can only be between 0
and 100
.
So with this on our mind, let's think of a new design for class Pokemon
:
class Pokemon:
"""Blueprint for a new pokemon"""
def __init__(self):
self._health = 100
# ^--- the leading _ is a convention to mark internal values
@property
def health(self):
"""The health of the Pokemon which is between 0 and 100"""
return self._health
@health.setter
def health(self, new_health):
"""Set the new heath value"""
# here the health limits are enforced
self._health = min(100, max(0, new_health))
def attack(self, other, choice):
"""Attack another Pokemon with the chosen attack (1 or 2)
This function also returns the raw amount of random damage dealt. The
amount of damage dealt depends on the attack type.
"""
if choice == 1:
attack_points = random.randint(18, 25)
elif choice == 2:
attack_points = random.randint(10, 35)
else:
print("That is not a selection. You lost your turn!")
attack_points = 0
other.health -= attack_points
return attack_points
def heal(self):
"""Heal the Pokemon"""
heal_points = random.randint(18, 35)
self.health += heal_points
return heal_points
A lot of this should look familiar to you, since the main work is still done by the code you've written. What is new that the Pokemon now has health. If your unfamiliar with properties in Python just think of them as synthatic sugar on what would normally be done using getter and setter functions in other languages. There is a great SO post with a nice explanation on properties. In addition to that, attack
and heal
now handle the update of the health value for the Pokemon, as well as the opponent. This allows us to write up a simple battle in a very concise way:
mew = Pokemon()
user = Pokemon()
mew.attack(user, 1)
print(f"User health after attack: user.health")
user.heal()
print(f"User health after heal: user.health")
mew.heal() # health should still only be 100
print(f"Mew's health after 'illegal' heal: user.health")
which prints for example:
User health after attack: 75
User health after heal: 100
Mew's health after 'illegal' heal: 100
No additional variables that need to track health status, no checking on the health limits. Everything is nicely encapsulated into the Pokemon
class.
Short break on style and other conventions
Another thing that has changed in contrast to your solution is the documentation I added to each function. Python has an "official" Style Guide (which is worth a read on its own) with a section on how to write these so called docstrings here.
It also features guidelines on where to use blank lines and where not to use them. In my oppinion the excessive use of blank lines in your code hinders readability more than it does help to structure the code.
I also used only a single leading underscore for my internal value instead of two as you did. The Style Guide also has you covered on this topic. In general you should always use a single leading underscore to mark functions and variables/members as internal. For more details on what happens if you use two leading underscores, follow the link above.
After that short intermezzo, let's look on how the battle simulation looks with the new class:
def battle_simulation():
"""Run a simple interactive Pokemon battle simulation"""
mew = Pokemon()
user_pokemon = Pokemon()
while True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = int(input("nSelect an attack: "))
# DON'T use eval on user input, this can be dangerous!
# Mew selects an attack, but focuses on attacking if health is full.
mew_choice = random.randint(1, 2 if mew.health == 100 else 3)
# this is your original distinction just condensed into a single line
# Attacks by user and Mew are done simultaneously
# with the changes to Pokemon, there is no need to save all the
# intermediate damage/heal values -> nice and short code
if attack_choice != 3:
print(f"You dealt user_pokemon.attack(mew, attack_choice) damage.")
if mew_choice != 3:
print(f"Mew dealt mew.attack(user_pokemon, mew_choice) damage.")
if attack_choice == 3:
print(f"You healed user_pokemon.heal() health points.")
if mew_choice == 3:
print(f"Mew healed mew.heal() health points.")
if mew.health == 0 or user_pokemon.health == 0:
break
print(f"Your current health is user_pokemon.health")
print(f"Mew's current health is mew.health")
print(f"Your final health is user_pokemon.health")
print(f"Mew's final health is mew.health")
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
if __name__ == "__main__":
battle_simulation()
As you can see, we got rid of all the variables needed to store damage and heal values since everything is no done internally in the Pokemon
class. With this, the code becomes quite a bit shorter, clearer and easier to read. Cool, isn't it?
A thing you will see me use quite often are the so called f-strings. They take arbitrary Python expressions (function calls, variables, ...) and allow to incorporate them into a formatted string. If you want to know more about them, I would recommend this blog post.
Happy Coding!
$endgroup$
add a comment |
$begingroup$
Conceptual
You have chosen a great problem to begin with, however there are a few things you get wrong about OOP. OOP is not simply about using a class for "everything", it's a kind of way to think about a problem.
To begin with, it helped me a lot to think about a class as a prototype of a "thing". In your case the "thing" would be a Pokemon. A Pokemon can do certain things. In your simplified versions that would be 1. attack another Pokemon and 2. heal itself. Often these actions are reflected in the classes's methods. I think you mostly understood that. What else is there about the Pokemon/"thing" it has certain properties that describe it. I would say that's an aspect you did not think about. A property could be name, color, ... or it's health status. We also learn that the health can only be between 0
and 100
.
So with this on our mind, let's think of a new design for class Pokemon
:
class Pokemon:
"""Blueprint for a new pokemon"""
def __init__(self):
self._health = 100
# ^--- the leading _ is a convention to mark internal values
@property
def health(self):
"""The health of the Pokemon which is between 0 and 100"""
return self._health
@health.setter
def health(self, new_health):
"""Set the new heath value"""
# here the health limits are enforced
self._health = min(100, max(0, new_health))
def attack(self, other, choice):
"""Attack another Pokemon with the chosen attack (1 or 2)
This function also returns the raw amount of random damage dealt. The
amount of damage dealt depends on the attack type.
"""
if choice == 1:
attack_points = random.randint(18, 25)
elif choice == 2:
attack_points = random.randint(10, 35)
else:
print("That is not a selection. You lost your turn!")
attack_points = 0
other.health -= attack_points
return attack_points
def heal(self):
"""Heal the Pokemon"""
heal_points = random.randint(18, 35)
self.health += heal_points
return heal_points
A lot of this should look familiar to you, since the main work is still done by the code you've written. What is new that the Pokemon now has health. If your unfamiliar with properties in Python just think of them as synthatic sugar on what would normally be done using getter and setter functions in other languages. There is a great SO post with a nice explanation on properties. In addition to that, attack
and heal
now handle the update of the health value for the Pokemon, as well as the opponent. This allows us to write up a simple battle in a very concise way:
mew = Pokemon()
user = Pokemon()
mew.attack(user, 1)
print(f"User health after attack: user.health")
user.heal()
print(f"User health after heal: user.health")
mew.heal() # health should still only be 100
print(f"Mew's health after 'illegal' heal: user.health")
which prints for example:
User health after attack: 75
User health after heal: 100
Mew's health after 'illegal' heal: 100
No additional variables that need to track health status, no checking on the health limits. Everything is nicely encapsulated into the Pokemon
class.
Short break on style and other conventions
Another thing that has changed in contrast to your solution is the documentation I added to each function. Python has an "official" Style Guide (which is worth a read on its own) with a section on how to write these so called docstrings here.
It also features guidelines on where to use blank lines and where not to use them. In my oppinion the excessive use of blank lines in your code hinders readability more than it does help to structure the code.
I also used only a single leading underscore for my internal value instead of two as you did. The Style Guide also has you covered on this topic. In general you should always use a single leading underscore to mark functions and variables/members as internal. For more details on what happens if you use two leading underscores, follow the link above.
After that short intermezzo, let's look on how the battle simulation looks with the new class:
def battle_simulation():
"""Run a simple interactive Pokemon battle simulation"""
mew = Pokemon()
user_pokemon = Pokemon()
while True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = int(input("nSelect an attack: "))
# DON'T use eval on user input, this can be dangerous!
# Mew selects an attack, but focuses on attacking if health is full.
mew_choice = random.randint(1, 2 if mew.health == 100 else 3)
# this is your original distinction just condensed into a single line
# Attacks by user and Mew are done simultaneously
# with the changes to Pokemon, there is no need to save all the
# intermediate damage/heal values -> nice and short code
if attack_choice != 3:
print(f"You dealt user_pokemon.attack(mew, attack_choice) damage.")
if mew_choice != 3:
print(f"Mew dealt mew.attack(user_pokemon, mew_choice) damage.")
if attack_choice == 3:
print(f"You healed user_pokemon.heal() health points.")
if mew_choice == 3:
print(f"Mew healed mew.heal() health points.")
if mew.health == 0 or user_pokemon.health == 0:
break
print(f"Your current health is user_pokemon.health")
print(f"Mew's current health is mew.health")
print(f"Your final health is user_pokemon.health")
print(f"Mew's final health is mew.health")
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
if __name__ == "__main__":
battle_simulation()
As you can see, we got rid of all the variables needed to store damage and heal values since everything is no done internally in the Pokemon
class. With this, the code becomes quite a bit shorter, clearer and easier to read. Cool, isn't it?
A thing you will see me use quite often are the so called f-strings. They take arbitrary Python expressions (function calls, variables, ...) and allow to incorporate them into a formatted string. If you want to know more about them, I would recommend this blog post.
Happy Coding!
$endgroup$
Conceptual
You have chosen a great problem to begin with, however there are a few things you get wrong about OOP. OOP is not simply about using a class for "everything", it's a kind of way to think about a problem.
To begin with, it helped me a lot to think about a class as a prototype of a "thing". In your case the "thing" would be a Pokemon. A Pokemon can do certain things. In your simplified versions that would be 1. attack another Pokemon and 2. heal itself. Often these actions are reflected in the classes's methods. I think you mostly understood that. What else is there about the Pokemon/"thing" it has certain properties that describe it. I would say that's an aspect you did not think about. A property could be name, color, ... or it's health status. We also learn that the health can only be between 0
and 100
.
So with this on our mind, let's think of a new design for class Pokemon
:
class Pokemon:
"""Blueprint for a new pokemon"""
def __init__(self):
self._health = 100
# ^--- the leading _ is a convention to mark internal values
@property
def health(self):
"""The health of the Pokemon which is between 0 and 100"""
return self._health
@health.setter
def health(self, new_health):
"""Set the new heath value"""
# here the health limits are enforced
self._health = min(100, max(0, new_health))
def attack(self, other, choice):
"""Attack another Pokemon with the chosen attack (1 or 2)
This function also returns the raw amount of random damage dealt. The
amount of damage dealt depends on the attack type.
"""
if choice == 1:
attack_points = random.randint(18, 25)
elif choice == 2:
attack_points = random.randint(10, 35)
else:
print("That is not a selection. You lost your turn!")
attack_points = 0
other.health -= attack_points
return attack_points
def heal(self):
"""Heal the Pokemon"""
heal_points = random.randint(18, 35)
self.health += heal_points
return heal_points
A lot of this should look familiar to you, since the main work is still done by the code you've written. What is new that the Pokemon now has health. If your unfamiliar with properties in Python just think of them as synthatic sugar on what would normally be done using getter and setter functions in other languages. There is a great SO post with a nice explanation on properties. In addition to that, attack
and heal
now handle the update of the health value for the Pokemon, as well as the opponent. This allows us to write up a simple battle in a very concise way:
mew = Pokemon()
user = Pokemon()
mew.attack(user, 1)
print(f"User health after attack: user.health")
user.heal()
print(f"User health after heal: user.health")
mew.heal() # health should still only be 100
print(f"Mew's health after 'illegal' heal: user.health")
which prints for example:
User health after attack: 75
User health after heal: 100
Mew's health after 'illegal' heal: 100
No additional variables that need to track health status, no checking on the health limits. Everything is nicely encapsulated into the Pokemon
class.
Short break on style and other conventions
Another thing that has changed in contrast to your solution is the documentation I added to each function. Python has an "official" Style Guide (which is worth a read on its own) with a section on how to write these so called docstrings here.
It also features guidelines on where to use blank lines and where not to use them. In my oppinion the excessive use of blank lines in your code hinders readability more than it does help to structure the code.
I also used only a single leading underscore for my internal value instead of two as you did. The Style Guide also has you covered on this topic. In general you should always use a single leading underscore to mark functions and variables/members as internal. For more details on what happens if you use two leading underscores, follow the link above.
After that short intermezzo, let's look on how the battle simulation looks with the new class:
def battle_simulation():
"""Run a simple interactive Pokemon battle simulation"""
mew = Pokemon()
user_pokemon = Pokemon()
while True:
print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
attack_choice = int(input("nSelect an attack: "))
# DON'T use eval on user input, this can be dangerous!
# Mew selects an attack, but focuses on attacking if health is full.
mew_choice = random.randint(1, 2 if mew.health == 100 else 3)
# this is your original distinction just condensed into a single line
# Attacks by user and Mew are done simultaneously
# with the changes to Pokemon, there is no need to save all the
# intermediate damage/heal values -> nice and short code
if attack_choice != 3:
print(f"You dealt user_pokemon.attack(mew, attack_choice) damage.")
if mew_choice != 3:
print(f"Mew dealt mew.attack(user_pokemon, mew_choice) damage.")
if attack_choice == 3:
print(f"You healed user_pokemon.heal() health points.")
if mew_choice == 3:
print(f"Mew healed mew.heal() health points.")
if mew.health == 0 or user_pokemon.health == 0:
break
print(f"Your current health is user_pokemon.health")
print(f"Mew's current health is mew.health")
print(f"Your final health is user_pokemon.health")
print(f"Mew's final health is mew.health")
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against Mew!")
if __name__ == "__main__":
battle_simulation()
As you can see, we got rid of all the variables needed to store damage and heal values since everything is no done internally in the Pokemon
class. With this, the code becomes quite a bit shorter, clearer and easier to read. Cool, isn't it?
A thing you will see me use quite often are the so called f-strings. They take arbitrary Python expressions (function calls, variables, ...) and allow to incorporate them into a formatted string. If you want to know more about them, I would recommend this blog post.
Happy Coding!
answered 5 hours ago
AlexAlex
1,383419
1,383419
add a comment |
add a comment |
$begingroup$
1) The doc comment in the class should give information for that class and not for the whole program.
2) I don't see why the Pokemon should be initialised with an attack choice if it can do multiple attacks. I think the choice should be passed as a parameter to the attack method. Instead it will make more sense for the class to hold the health and the logic for the damage which is currently in the while
loop.
3) battle_continue == True
can be simplified to only battle_continue
and can be renamed to battle_in_progress
.
4) The print statement: print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
can be separated into multiple lines for better readability.
5) The moves can be an Enum
instead of magic numbers.
6) The move selection should have exception handling for invalid inputs.
7) The attack function can take as a parameter the target of the attack and deduce the life points.
8) Some of the other functionality is better suited as methods on the class e.g checking if the health is 0, applying damage to a Pokemon, healing.
Final Implementation:
import random
from enum import Enum
class Pokemon:
"""Class for representing a pokemon"""
def __init__(self, name):
self.name = name
self.health = 100
def make_move(self, move, target):
if move == Moves.HEAL:
self.heal()
else:
self.attack(move, target)
def attack(self, move, target):
attack_points = None
if move == Moves.CLOSE_ATTACK:
attack_points = random.randint(18, 25)
if move == Moves.RANGE_ATTACK:
attack_points = random.randint(10, 35)
if attack_points:
target.take_damage(attack_points)
print(self.name, "dealt", attack_points, "damage.")
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18, 35)
self.health = self.health + heal_points
print(self.name, "healed", heal_points, "health points.")
if self.health > 100:
self.health = 100
def take_damage(self, points):
self.health = self.health - points
if self.health < 0:
self.health = 0
def is_dead(self):
return self.health <= 0
class PokemonBot(Pokemon):
def select_move(self):
if self.health == 100:
choice = Moves.get_random_attack()
else:
choice = Moves.get_random_move()
return choice
class Moves(Enum):
CLOSE_ATTACK = 1
RANGE_ATTACK = 2
HEAL = 3
@staticmethod
def get_random_attack():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK])
@staticmethod
def get_random_move():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK, Moves.HEAL])
###########################################################################
battle_in_progress = True
mew = PokemonBot("Mew")
name = input("Name your pokemon: ")
user_pokemon = Pokemon(name)
while battle_in_progress:
print("MOVE CHOICES")
print("1. Close range attack")
print("2. Far range attack")
print("3. Heal")
move_choice = None
while move_choice is None:
try:
move_choice = Moves(int(input("Select a move: ")))
except ValueError:
print("Please enter a valid move!")
mew_move = mew.select_move()
user_pokemon.make_move(move_choice, mew)
mew.make_move(mew_move, user_pokemon)
print("Current health of", user_pokemon.name, "is", user_pokemon.health)
print("Current health of", mew.name, "is", mew.health)
if mew.is_dead() or user_pokemon.is_dead():
battle_in_progress = False
print("Final health of", user_pokemon.name, "is", user_pokemon.health)
print("Final health of", mew.name, "is", mew.health)
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against", mew.name, "!")
$endgroup$
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
add a comment |
$begingroup$
1) The doc comment in the class should give information for that class and not for the whole program.
2) I don't see why the Pokemon should be initialised with an attack choice if it can do multiple attacks. I think the choice should be passed as a parameter to the attack method. Instead it will make more sense for the class to hold the health and the logic for the damage which is currently in the while
loop.
3) battle_continue == True
can be simplified to only battle_continue
and can be renamed to battle_in_progress
.
4) The print statement: print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
can be separated into multiple lines for better readability.
5) The moves can be an Enum
instead of magic numbers.
6) The move selection should have exception handling for invalid inputs.
7) The attack function can take as a parameter the target of the attack and deduce the life points.
8) Some of the other functionality is better suited as methods on the class e.g checking if the health is 0, applying damage to a Pokemon, healing.
Final Implementation:
import random
from enum import Enum
class Pokemon:
"""Class for representing a pokemon"""
def __init__(self, name):
self.name = name
self.health = 100
def make_move(self, move, target):
if move == Moves.HEAL:
self.heal()
else:
self.attack(move, target)
def attack(self, move, target):
attack_points = None
if move == Moves.CLOSE_ATTACK:
attack_points = random.randint(18, 25)
if move == Moves.RANGE_ATTACK:
attack_points = random.randint(10, 35)
if attack_points:
target.take_damage(attack_points)
print(self.name, "dealt", attack_points, "damage.")
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18, 35)
self.health = self.health + heal_points
print(self.name, "healed", heal_points, "health points.")
if self.health > 100:
self.health = 100
def take_damage(self, points):
self.health = self.health - points
if self.health < 0:
self.health = 0
def is_dead(self):
return self.health <= 0
class PokemonBot(Pokemon):
def select_move(self):
if self.health == 100:
choice = Moves.get_random_attack()
else:
choice = Moves.get_random_move()
return choice
class Moves(Enum):
CLOSE_ATTACK = 1
RANGE_ATTACK = 2
HEAL = 3
@staticmethod
def get_random_attack():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK])
@staticmethod
def get_random_move():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK, Moves.HEAL])
###########################################################################
battle_in_progress = True
mew = PokemonBot("Mew")
name = input("Name your pokemon: ")
user_pokemon = Pokemon(name)
while battle_in_progress:
print("MOVE CHOICES")
print("1. Close range attack")
print("2. Far range attack")
print("3. Heal")
move_choice = None
while move_choice is None:
try:
move_choice = Moves(int(input("Select a move: ")))
except ValueError:
print("Please enter a valid move!")
mew_move = mew.select_move()
user_pokemon.make_move(move_choice, mew)
mew.make_move(mew_move, user_pokemon)
print("Current health of", user_pokemon.name, "is", user_pokemon.health)
print("Current health of", mew.name, "is", mew.health)
if mew.is_dead() or user_pokemon.is_dead():
battle_in_progress = False
print("Final health of", user_pokemon.name, "is", user_pokemon.health)
print("Final health of", mew.name, "is", mew.health)
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against", mew.name, "!")
$endgroup$
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
add a comment |
$begingroup$
1) The doc comment in the class should give information for that class and not for the whole program.
2) I don't see why the Pokemon should be initialised with an attack choice if it can do multiple attacks. I think the choice should be passed as a parameter to the attack method. Instead it will make more sense for the class to hold the health and the logic for the damage which is currently in the while
loop.
3) battle_continue == True
can be simplified to only battle_continue
and can be renamed to battle_in_progress
.
4) The print statement: print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
can be separated into multiple lines for better readability.
5) The moves can be an Enum
instead of magic numbers.
6) The move selection should have exception handling for invalid inputs.
7) The attack function can take as a parameter the target of the attack and deduce the life points.
8) Some of the other functionality is better suited as methods on the class e.g checking if the health is 0, applying damage to a Pokemon, healing.
Final Implementation:
import random
from enum import Enum
class Pokemon:
"""Class for representing a pokemon"""
def __init__(self, name):
self.name = name
self.health = 100
def make_move(self, move, target):
if move == Moves.HEAL:
self.heal()
else:
self.attack(move, target)
def attack(self, move, target):
attack_points = None
if move == Moves.CLOSE_ATTACK:
attack_points = random.randint(18, 25)
if move == Moves.RANGE_ATTACK:
attack_points = random.randint(10, 35)
if attack_points:
target.take_damage(attack_points)
print(self.name, "dealt", attack_points, "damage.")
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18, 35)
self.health = self.health + heal_points
print(self.name, "healed", heal_points, "health points.")
if self.health > 100:
self.health = 100
def take_damage(self, points):
self.health = self.health - points
if self.health < 0:
self.health = 0
def is_dead(self):
return self.health <= 0
class PokemonBot(Pokemon):
def select_move(self):
if self.health == 100:
choice = Moves.get_random_attack()
else:
choice = Moves.get_random_move()
return choice
class Moves(Enum):
CLOSE_ATTACK = 1
RANGE_ATTACK = 2
HEAL = 3
@staticmethod
def get_random_attack():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK])
@staticmethod
def get_random_move():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK, Moves.HEAL])
###########################################################################
battle_in_progress = True
mew = PokemonBot("Mew")
name = input("Name your pokemon: ")
user_pokemon = Pokemon(name)
while battle_in_progress:
print("MOVE CHOICES")
print("1. Close range attack")
print("2. Far range attack")
print("3. Heal")
move_choice = None
while move_choice is None:
try:
move_choice = Moves(int(input("Select a move: ")))
except ValueError:
print("Please enter a valid move!")
mew_move = mew.select_move()
user_pokemon.make_move(move_choice, mew)
mew.make_move(mew_move, user_pokemon)
print("Current health of", user_pokemon.name, "is", user_pokemon.health)
print("Current health of", mew.name, "is", mew.health)
if mew.is_dead() or user_pokemon.is_dead():
battle_in_progress = False
print("Final health of", user_pokemon.name, "is", user_pokemon.health)
print("Final health of", mew.name, "is", mew.health)
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against", mew.name, "!")
$endgroup$
1) The doc comment in the class should give information for that class and not for the whole program.
2) I don't see why the Pokemon should be initialised with an attack choice if it can do multiple attacks. I think the choice should be passed as a parameter to the attack method. Instead it will make more sense for the class to hold the health and the logic for the damage which is currently in the while
loop.
3) battle_continue == True
can be simplified to only battle_continue
and can be renamed to battle_in_progress
.
4) The print statement: print("nATTACK CHOICESn1. Close range attackn2. Far range attackn3. Heal")
can be separated into multiple lines for better readability.
5) The moves can be an Enum
instead of magic numbers.
6) The move selection should have exception handling for invalid inputs.
7) The attack function can take as a parameter the target of the attack and deduce the life points.
8) Some of the other functionality is better suited as methods on the class e.g checking if the health is 0, applying damage to a Pokemon, healing.
Final Implementation:
import random
from enum import Enum
class Pokemon:
"""Class for representing a pokemon"""
def __init__(self, name):
self.name = name
self.health = 100
def make_move(self, move, target):
if move == Moves.HEAL:
self.heal()
else:
self.attack(move, target)
def attack(self, move, target):
attack_points = None
if move == Moves.CLOSE_ATTACK:
attack_points = random.randint(18, 25)
if move == Moves.RANGE_ATTACK:
attack_points = random.randint(10, 35)
if attack_points:
target.take_damage(attack_points)
print(self.name, "dealt", attack_points, "damage.")
else:
print("That is not a selection. You lost your turn!")
def heal(self):
heal_points = random.randint(18, 35)
self.health = self.health + heal_points
print(self.name, "healed", heal_points, "health points.")
if self.health > 100:
self.health = 100
def take_damage(self, points):
self.health = self.health - points
if self.health < 0:
self.health = 0
def is_dead(self):
return self.health <= 0
class PokemonBot(Pokemon):
def select_move(self):
if self.health == 100:
choice = Moves.get_random_attack()
else:
choice = Moves.get_random_move()
return choice
class Moves(Enum):
CLOSE_ATTACK = 1
RANGE_ATTACK = 2
HEAL = 3
@staticmethod
def get_random_attack():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK])
@staticmethod
def get_random_move():
return random.choice([Moves.CLOSE_ATTACK, Moves.RANGE_ATTACK, Moves.HEAL])
###########################################################################
battle_in_progress = True
mew = PokemonBot("Mew")
name = input("Name your pokemon: ")
user_pokemon = Pokemon(name)
while battle_in_progress:
print("MOVE CHOICES")
print("1. Close range attack")
print("2. Far range attack")
print("3. Heal")
move_choice = None
while move_choice is None:
try:
move_choice = Moves(int(input("Select a move: ")))
except ValueError:
print("Please enter a valid move!")
mew_move = mew.select_move()
user_pokemon.make_move(move_choice, mew)
mew.make_move(mew_move, user_pokemon)
print("Current health of", user_pokemon.name, "is", user_pokemon.health)
print("Current health of", mew.name, "is", mew.health)
if mew.is_dead() or user_pokemon.is_dead():
battle_in_progress = False
print("Final health of", user_pokemon.name, "is", user_pokemon.health)
print("Final health of", mew.name, "is", mew.health)
if user_pokemon.health < mew.health:
print("nYou lost! Better luck next time!")
else:
print("nYou won against", mew.name, "!")
edited 5 hours ago
answered 5 hours ago
DobromirMDobromirM
1767
1767
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
add a comment |
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
1
1
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
This is more of a competing implementation than a review. Please take OP through what needs changing from their code, and why.
$endgroup$
– Austin Hastings
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
$begingroup$
I've added some more information, I just wanted to post the code before I started improving the answer.
$endgroup$
– DobromirM
5 hours ago
add a comment |
yummylipbalm is a new contributor. Be nice, and check out our Code of Conduct.
yummylipbalm is a new contributor. Be nice, and check out our Code of Conduct.
yummylipbalm is a new contributor. Be nice, and check out our Code of Conduct.
yummylipbalm is a new contributor. Be nice, and check out our Code of Conduct.
Thanks for contributing an answer to Code Review Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
Use MathJax to format equations. MathJax reference.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f217222%2fpokemon-turn-based-battle-python%23new-answer', 'question_page');
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
1
$begingroup$
Welcome to Code Review! I hope you get some great answers. Please include a (possible shortened) description of the task you want to accomplish in case the external description is somehow no longer available.
$endgroup$
– Alex
6 hours ago
$begingroup$
I suspect you have a typo, as the code presented doesn't follow the spec:
heal_points = random.randint(18,35)
does not have the same range as attack option one, topping out at 35 instead of 25$endgroup$
– TemporalWolf
5 hours ago
$begingroup$
OOP, as it is typically practiced in languages like C# and Java, is not strongly encouraged in Python. Python's late binding on all names (include modules, classes, and unattached methods) negates many of the benefits of pervasive object use in other languages. More procedurally or functionally minded approaches are more common and often (or maybe usually) simpler.
$endgroup$
– jpmc26
4 hours ago