I'm new to python and programming in general, so would really appreciate any clarification on this point.
For example, in the following code:
#Using a class
class Monster(object):
def __init__(self, level, damage, duration):
print self
self.level = level
self.damage = damage
self.duration = duration
def fight(self):
print self
print "The monster's level is ", self.level
print "The monster's damage is ", self.damage
print "The attack duration is ", self.duration
def sleep(self):
print "The monster is tired and decides to rest."
x = Monster(1, 2, 3)
y = Monster(2, 3, 4)
x.fight()
y.fight()
y.sleep()
#just using a function
def fight_func(level, damage, duration):
print "The monster's level is ", level
print "The monster's damage is ", damage
print "The attack duration is ", duration
fight_func(1, 2, 3)
fight_func(5,3,4)
The pure function version seems cleaner and gives the same result.
Is the primary value of classes that you can create and call a number of methods on an object, e.g. fight or sleep?
This is a more general answer than for just your code, but classes are useful when you need to:
Share state safely across specific functions.
Classes offer encapsulation to avoid polluting the global namespace with variables that only need to be shared across a few functions. You don't want someone using your code to overwrite a variable all of your functions are using - you always want to be able to control access as much as possible.
Classes can be instantiated and have multiple independent copies of the same variable that are not shared at all while requiring very little code use.
They're appropriate when you need to model interactions between a collection of functions and variables with another collection of functions and variables i.e. when you need to model objects. Defining a class for a single function with no state is frankly a waste of code.
Provide operator overloading or define specific traits for certain types
What makes
int
,string
, andtuples
hashable (can be used as keys in a dictionary or inserted into aset
), whilelist
s anddict
s aren't?Answer: the base classes for these primitive types implement the magic method
__hash__
, which enables the Python compiler to dynamically compute a hash at runtime.What allows the
+
operator to be overloaded, so that5 + 2 = 7
forint
s but's' + 'a' = 'sa'
for strings?Answer: the base classes for these types defines their own
__add__
method, which lets you define exactly how addition between two members of your class is carried out.All user-defined classes can implement these methods. There are good use cases for them: larger mathematical libraries - numpy, scipy, and pandas - make heavy use of operator overloading by defining these magic methods to provide you useful objects to handle your data. Classes are the only way to implement these traits and properties in Python.
Classes are not superior to functions when the above conditions are not met.
Use a class to group together meaningful functions and variables in a sensible way, or to endow your objects with special properties. Anything else is superfluous.
what about my monsters? he can fight another monster! Presumably your functional monster cannot do that ;-)
so:
Your example is rather simplified.
In a more complete example fighting wouldn't just display the current state - it would also modify that state. Your monster might get hurt and that would change its hit points and morale. This state has to be stored somewhere. If you use a class it would be natural to add instance variables to store this state.
Using only functions it would be more difficult to find a good place to store the state.
Your example isn't particularly interesting — it just prints three numbers. You don't need classes, or even a function (really) to do that.
But if you ever need to write a program that has to keep track of two separate monsters at one time, know their health, differentiate their fighting abilities, and so, you can see the value of encapsulating each of the monsters in a separate monster instance.