For many PHP programmers, object-oriented programming is a frightening concept, full of complicated syntax and other roadblocks. As detailed in my book, Pro PHP and jQuery, you'll learn the concepts behind object-oriented programming (OOP), a style of coding in which related actions are grouped into classes to aid in creating more-compact, effective code.
Understanding Object-Oriented Programming
Object-oriented programming is a style of coding that allows developers to group similar tasks into classes. This helps keep code following the tenet "don't repeat yourself" (DRY) and easy-to-maintain.
"Object-oriented programming is a style of coding that allows developers to group similar tasks into classes."
One of the major benefits of DRY programming is that, if a piece of information changes in your program, usually only one change is required to update the code. One of the biggest nightmares for developers is maintaining code where data is declared over and over again, meaning any changes to the program become an infinitely more frustrating game of Where's Waldo? as they hunt for duplicated data and functionality.
OOP is intimidating to a lot of developers because it introduces new syntax and, at a glance, appears to be far more complex than simple procedural, or inline, code. However, upon closer inspection, OOP is actually a very straightforward and ultimately simpler approach to programming.
Understanding Objects and Classes
Before you can get too deep into the finer points of OOP, a basic understanding of the differences between objects and classes is necessary. This section will go over the building blocks of classes, their different capabilities, and some of their uses.
Recognizing the Differences Between Objects and Classes
Developers start talking about objects and classes, and they appear to be interchangeable terms. This is not the case, however.
Right off the bat, there's confusion in OOP: seasoned developers start talking about objects and classes, and they appear to be interchangeable terms. This is not the case, however, though the difference can be tough to wrap your head around at first.
A class, for example, is like a blueprint for a house. It defines the shape of the house on paper, with relationships between the different parts of the house clearly defined and planned out, even though the house doesn't exist.
An object, then, is like the actual house built according to that blueprint. The data stored in the object is like the wood, wires, and concrete that compose the house: without being assembled according to the blueprint, it's just a pile of stuff. However, when it all comes together, it becomes an organized, useful house.
Classes form the structure of data and actions and use that information to build objects. More than one object can be built from the same class at the same time, each one independent of the others. Continuing with our construction analogy, it's similar to the way an entire subdivision can be built from the same blueprint: 150 different houses that all look the same but have different
families and decorations inside.
families and decorations inside.
Structuring Classes
The syntax to create a class is pretty straightforward: declare a class using the
class
keyword, followed by the name of the class and a set of curly braces ({}
):
1
2
3
4
5
6
7
8
| <?php class MyClass { // Class properties and methods go here } ?> |
After creating the class, a new class can be instantiated and stored in a variable using the
new
keyword:
1
| $obj = new MyClass; |
To see the contents of the class, use
var_dump()
:
1
| var_dump( $obj ); |
Try out this process by putting all the preceding code in a new file called
test.php
in [your local] testing folder:
01
02
03
04
05
06
07
08
09
10
11
12
| <?php class MyClass { // Class properties and methods go here } $obj = new MyClass; var_dump( $obj ); ?> |
Load the page in your browser at
http://localhost/test.php
and the following should display:
1
| object(MyClass)#1 (0) { } |
In its simplest form, you've just completed your first OOP script.
Defining Class Properties
To add data to a class, properties, or class-specific variables, are used. These work exactly like regular variables, except they're bound to the object and therefore can only be accessed using the object.
To add a property to
MyClass
, add the following code to your script:
01
02
03
04
05
06
07
08
09
10
11
12
| <?php class MyClass { public $prop1 = "I'm a class property!" ; } $obj = new MyClass; var_dump( $obj ); ?> |
The keyword
public
determines the visibility of the property, which you'll learn about a little later in this chapter. Next, the property is named using standard variable syntax, and a value is assigned (though class properties do not need an initial value).
To read this property and output it to the browser, reference the object from which to read and the property to be read:
1
| echo $obj ->prop1; |
Because multiple instances of a class can exist, if the individual object is not referenced, the script would be unable to determine which object to read from. The use of the arrow (
->
) is an OOP construct that accesses the contained properties and methods of a given object.
Modify the script in
test.php
to read out the property rather than dumping the whole class by modifying the code as shown:
01
02
03
04
05
06
07
08
09
10
11
12
| <?php class MyClass { public $prop1 = "I'm a class property!" ; } $obj = new MyClass; echo $obj ->prop1; // Output the property ?> |
Reloading your browser now outputs the following:
1
| I'm a class property! |
Defining Class Methods
Methods are class-specific functions. Individual actions that an object will be able to perform are defined within the class as methods.
For instance, to create methods that would set and get the value of the class property
$prop1
, add the following to your code:
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } $obj = new MyClass; echo $obj ->prop1; ?> |
Note — OOP allows objects to reference themselves using
$this
. When working within a method, use $this
in the same way you would use the object name outside the class.
To use these methods, call them just like regular functions, but first, reference the object they belong to. Read the property from
MyClass
, change its value, and read it out again by making the modifications below:
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } $obj = new MyClass; echo $obj ->getProperty(); // Get the property value $obj ->setProperty( "I'm a new property value!" ); // Set a new one echo $obj ->getProperty(); // Read it out again to show the change ?> |
Reload your browser, and you'll see the following:
1
2
| I'm a class property! I'm a new property value! |
"The power of OOP becomes apparent when using multiple instances of the
same class."
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } // Create two objects $obj = new MyClass; $obj2 = new MyClass; // Get the value of $prop1 from both objects echo $obj ->getProperty(); echo $obj2 ->getProperty(); // Set new values for both objects $obj ->setProperty( "I'm a new property value!" ); $obj2 ->setProperty( "I belong to the second instance!" ); // Output both objects' $prop1 value echo $obj ->getProperty(); echo $obj2 ->getProperty(); ?> |
When you load the results in your browser, they read as follows:
1
2
3
4
| I'm a class property! I'm a class property! I'm a new property value! I belong to the second instance! |
As you can see, OOP keeps objects as separate entities, which makes for easy separation of different pieces of code into small, related bundles.
Magic Methods in OOP
To make the use of objects easier, PHP also provides a number of magic methods, or special methods that are called when certain common actions occur within objects. This allows developers to perform a number of useful tasks with relative ease.
Using Constructors and Destructors
When an object is instantiated, it's often desirable to set a few things right off the bat. To handle this, PHP provides the magic method
created.
__construct()
, which is called automatically whenever a new object iscreated.
For the purpose of illustrating the concept of constructors, add a constructor to
MyClass
that will output a message whenever a new instance of the class is created:
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function __construct() { echo 'The class "' , __CLASS__ , '" was initiated!<br />' ; } public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } // Create a new object $obj = new MyClass; // Get the value of $prop1 echo $obj ->getProperty(); // Output a message at the end of the file echo "End of file.<br />" ; ?> |
Note —
__CLASS__
returns the name of the class in which it is called; this is what is known as a magic constant. There are several available magic constants, which you can read more about in the PHP manual.
Reloading the file in your browser will produce the following result:
1
2
3
| The class "MyClass" was initiated! I'm a class property! End of file. |
To call a function when the object is destroyed, the
__destruct()
magic method is available. This is useful for class cleanup (closing a database connection, for instance).
Output a message when the object is destroyed by defining the magic method
__destruct()
in MyClass
:
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function __construct() { echo 'The class "' , __CLASS__ , '" was initiated!<br />' ; } public function __destruct() { echo 'The class "' , __CLASS__ , '" was destroyed.<br />' ; } public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } // Create a new object $obj = new MyClass; // Get the value of $prop1 echo $obj ->getProperty(); // Output a message at the end of the file echo "End of file.<br />" ; ?> |
With a destructor defined, reloading the test file results in the following output:
1
2
3
4
| The class "MyClass" was initiated! I'm a class property! End of file. The class "MyClass" was destroyed. |
"When the end of a file is reached, PHP automatically releases all resources."
To explicitly trigger the destructor, you can destroy the object using the
function
function
unset()
:
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
| <?php class MyClass { public $prop1 = "I'm a class property!" ; public function __construct() { echo 'The class "' , __CLASS__ , '" was initiated!<br />' ; } public function __destruct() { echo 'The class "' , __CLASS__ , '" was destroyed.<br />' ; } public function setProperty( $newval ) { $this ->prop1 = $newval ; } public function getProperty() { return $this ->prop1 . "<br />" ; } } // Create a new object $obj = new MyClass; // Get the value of $prop1 echo $obj ->getProperty(); // Destroy the object unset( $obj ); // Output a message at the end of the file echo "End of file.<br />" ; ?> |
Now the result changes to the following when loaded in your browser:
1
2
3
4
| The class "MyClass" was initiated! I'm a class property! The class "MyClass" was destroyed. End of file. |
Converting to a String
To avoid an error if a script attempts to output
MyClass
as a string, another magic method is used called __toString()
.
No comments:
Post a Comment