Defining Classes
Defining a custom class is an extremely useful way to use Python. A class object has two kinds of class attributes: data attributes that store data and methods, which, as we have already seen, are functions that belong to and often operate on instances of an object.
A custom class is a convenient way to represent many kinds of objects in engineering. Here are some examples with potential data attributes and methods included:
- A time-varying signal class with data attributes
periodic
,period
,amplitude
, andfrequency
and methodsrms()
,abs()
, andplot()
- An experiment simulation class with data attributes
time
,executed
,input
, andoutput
and methodsexecute()
,plot()
, andsave()
- A truss class with data attributes
members
,connections
,pin_angles
,member_forces
, andreactions
and methodsanalyze()
,max_compression()
,max_tension()
, andmax_reaction()
The basic syntax for a class definition is as follows:
class ClassName:
"""Docsting description"""
<Statement 1>
<Statement 2>
<etc.>
Data attributes can be defined via the usual variable assignment syntax and generally follow the docstring. Method definitions follow data attributes. Consider the following class definition to represent a screwdriver tool (perhaps in the context of a robot’s inventory of available tools):
class Screwdriver:
"""Represents a screwdriver tool"""
= "Screw" # Class data attributes
operates_on = "Hand"
operated_by
def drive(self, screw, angle): # Method definition
"""Returns a screw object turned by the given angle"""
return screw.turn(angle)
Any object that is an instance of the class Screwdriver
will have the class
attributes defined above. To create an instance (i.e., instantiate),
call the class name as if it were a function with no arguments, as
follows:
= Screwdriver() # Create an instance of the Screwdriver class
sd1 = Screwdriver() # Another instance
sd2 # Access class attributes
sd1.operates_on sd1.operated_by
'Screw'
'Hand'
In many cases, we will define a special constructor method named __init__()
,
which will be called at instatiation and passed any arguments provided
as follows (we remove docstrings for brevity):
class Screwdriver:
= "Screw" # Class data attributes
operates_on = "Hand"
operated_by
def __init__(self, head, length):
self.head = head # Instance data attributes
self.length = length
def drive(self, screw, angle): # Method definition
return screw.turn(angle)
The attributes assigned to self
in the
__init__()
method are called instance data attributes.
The arguments head
and length
are required positional
arguments that are assigned to the instance data attributes head
and length
.
Consider the following instances:
= Screwdriver(head="Phillips", length=7)
sd1 = Screwdriver(head="Flat", length=8)
sd2 print(f"sd1 is a {sd1.head}head operated by {sd1.operated_by}")
print(f"sd2 is a {sd2.head}head operated by {sd2.operated_by}")
sd1 is a Phillipshead operated by Hand
sd2 is a Flathead operated by Hand
So we see that instances can have different instance data attributes but they share the same class data attributes.
Note that every method has as its first argument self
, which is
the conventional name given to the first argument, which is always the
instance object that includes the method. When calling a method of an
instance, we do not provide the self
argument
because it is provided automatically. Before we can call the Screwdriver
method drive()
, we should define a Screw
class as follows:
class Screw:
"""Represents a screw fastener"""
def __init__(self, head, angle=0, handed="Right"):
self.head = head
self.angle = angle
self.handed = handed
def turn(self, angle):
"""Mutates angle attribute by adding angle"""
self.angle += angle
Instances of the Screw
class
have \(3\) instance attributes, head
, angle
, and handed
. Let’s instantiate a screw and
give it a turn as follows:
= Screw(head="Phillips")
s1 print(f"Initial angle: {s1.angle}")
=s1, angle=3) # Turn the screw 3 units
sd1.drive(screwprint(f"Mutated angle: {s1.angle}")
=s1, angle=6) # Turn the screw 6 units
sd1.drive(screwprint(f"Mutated angle: {s1.angle}")
Initial angle: 0
Mutated angle: 3
Mutated angle: 9
As we have seen in this example, instance data attributes can represent the state of an object and methods can mutate or transition that state. This opens up a vast number of possibilities for the engineer, for we often need to keep track of the states and state transitions of objects in engineering systems.
Derived Classes
A derived class (also called a subclass) is a class that uses another class as its basis. A class that is a derived class’s basis is called a base class for the derived class. A derived class inherits all of the class data attributes and methods of its base class, and it typically has additional class data attributes or methods of its own.
Continuing the screw and screwdriver example from above, let’s define a derived class for representing set screws1 as follows:
class SetScrew(Screw):
"""Represents a set screw fastener"""
def __init__(self, head, tip, angle=0, handed="Right"):
self.tip = tip # Add instance attribute
super().__init__(head, angle, handed) # Call base constructor
The base class was specified by placing it in parentheses in SetScrew(Screw)
. It is not necessary to
define a new constructor, but to we did so to add the instance attribute
tip
. Rather than duplicating the
rest of the base class’s constructor, the base constructor was called
via super().__init__()
,
to which its relevant arguments were passed straight through. Trying out
the subclass,
= Screwdriver(head="Hex", length=5)
sd3 = SetScrew(head="Hex", tip="Nylon")
ss1 2) # Drive the set screw
sd3.drive(ss1, print(f"Set screw angle: {ss1.angle}")
Set screw angle: 2
Note what has occurred: the Screwdriver
instance sd3
used its drive()
method to call the turn()
method of SetScrew
instance ss1
. This turn()
method was inherited from the
Screw
class, so we didn’t have to
repeat the definition of turn()
in the subclass definition of SetScrew
.
A set screw is a screw that holds an object in place via a force applied by its tip.↩︎
Online Resources for Section 2.5
No online resources.