Object-Oriented PHP for Beginners

Drupalers

James Candan /jɑnˈdən/

7 years PHP

5 years Drupal

 

Object-Oriented PHP for Beginners | Tuts+
DEC 23, 2011 / JASON LENGSTORF

http://code.tutsplus.com/tutorials/object-oriented-php-for-beginners--net-12762

 

Prerequisites

<?php

// Variables
$fruit = "apple";
  • Variables
  • Functions
  • Arrays/Loops
  • Logic
<?php

// Variables
$fruit = "apple";

// Logic operators
if ( $fruit == "apple" ) { 
    // can include html in string output
    echo "sweeeeeet!<br />"; 
}

// Arrays
$colors = array("blue", "purple", 'red');

// Loops
foreach ($colors as $color) {
    echo $color . "<br />";
}

<?php

// Variables
$fruit = "apple";

// Logic operators
if ( $fruit == "apple" ) { 
    // can include html in string output
    echo "sweeeeeet!<br />"; 
}

<?php

// Variables
$fruit = "apple";

// Logic operators
if ( $fruit == "apple" ) { 
    // can include html in string output
    echo "sweeeeeet!<br />"; 
}

// Arrays
$colors = array("blue", "purple", 'red');

// Loops
foreach ($colors as $color) {
    echo $color . "<br />";
}

// Functions
function color_exists($needle, $haystack) {
    return in_array($needle, $haystack);
}

// assign returned output to variable
$is_brown_good = color_exists("brown", $colors);

Basics

You might already...

Object properties

<?php

$node['type'];
$node['title'];

Not the same as an Array.

<?php

$node = node_load($nid);
echo $node->type;
echo $node->title;
class User {
  ...

  public function run($miles) {
    $this->miles += $miles;
  }
}

$user = new User();
$user->run($miles);

Consider this:

Objectives

  • High level overview
  • Define common terms
  • Review practical examples

Understanding OOP

Style

  • Group similar code into CLASSES
  • DRY - Don't Repeat Yourself
  • Simple, Straightforward Structure

"Object-oriented programming is a style of coding that allows developers to group similar tasks into classes."

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.

Developers start talking about objects and classes, and they appear to be interchangeable terms.

 This is not the case.

Recognizing the Differences Between Objects and Classes

  • Classes are like blue prints
  • Objects are the actual Houses

Properties

Height

Methods

Come

Property values

Color: Gray

Eye Color: Brown

Height: 18"

Length: 36"

Weight: 30 lbs

Methods

Sit

Lay Down

Shake

Come

Create Instance

Object-Oriented Thinking
JULY 16, 2014 / RODRIGO ARAÚJO

http://www.universocomputacao.com/object-oriented-thinking/

Class

Object

Length

Weight

Eye Color

Color

Shake

Lay Down

Sit

Maximus

Structure

Structuring Classes

<?php

class Dog
{
    // class properties
    public $color;
    public $eye_color;
    public $height;
    public $length;
    public $weight;

    // class methods
    public function sit() {

    }
    public function lay_down() {

    }
    public function shake() {

    }
    public function come() {

    }
}
<?php

$maximus = new Dog();
$maximus->color     = "Gray";
$maximus->eye_color = "Brown";
$maximus->height    = 18;
$maximus->length    = 36;
$maximus->weight    = 30;

?>

Defining Class Properties

<?php
 
class Person
{
    // Let's write our first property
    public $name = "john";
}
<?php
 
class Person
{
    // Let's write our first property
    public $name = "john";
    public $test = "blue";
}

$my_person = new Person();
<?php
 
class Person
{

}
<?php
 
class Person
{
    // Let's write our first property
    public $name = "john";
    public $test = "blue";
}

$my_person = new Person();


var_dump($my_person);


echo $my_person->name;

Defining Class Methods

<?php
 
class Person
{
    public $name;
    public $weight = 0;
    public $miles = 0; 






}

$my_person = new Person();
$my_person->name = "john";
$my_person->weight = 190;

// notice: weight is 190
var_dump($my_person);
<?php
 
class Person
{
    public $name;
    public $weight = 0;
    public $miles = 0;   

    // Let's write our first method
    public function run($miles) {
        $this->miles += $miles;
        $this->weight -= $miles * .2;
    }
}

$my_person = new Person();
$my_person->name = "john";
$my_person->weight = 190;

// notice: weight is 190
var_dump($my_person);
<?php
 
class Person
{
    public $name;
    public $weight = 0;
    public $miles = 0;  

    // Let's write our first method
    public function run($miles) {
        $this->miles += $miles;
        $this->weight -= $miles * .2;
    }
}

$my_person = new Person();
$my_person->name = "john";
$my_person->weight = 190;

// notice: weight is 190
var_dump($my_person);

// notice: we've lost weight from running
$my_person->run(10);
var_dump($my_person);

PHP vs Javascript

<?php
 
class Person
{
    public $name;
    public $weight = 0;
    public $miles = 0;  

    // Let's write our first method
    public function run($miles) {
        $this->miles += $miles;
        $this->weight -= $miles;
    }
}

$my_person = new Person();
$my_person->name = "john";
$my_person->weight = 170;

// notice: weight is 170
var_dump($my_person);

// notice: we've lost weight from running
$my_person->run(5);
var_dump($my_person);

<script type="text/javascript">

var Person = function () {
    this.name = '';
    this.weight = 0;
    this.miles = 0;
};

Person.prototype.run = function (miles) {
    this.miles += miles;
    this.weight -= miles;
};

var my_person = new Person();
my_person.name = "john";
my_person.weight = 170;

console.log(my_person.name); // john
console.log(my_person.weight); // 170
console.log(my_person.miles); // 0

my_person.run(5);
console.log(my_person.name); // john
console.log(my_person.weight); // 165
console.log(my_person.miles); // 5

</script>

Magic Methods

<?php

class Dog {
        
  public $legs;
  public $color;
  public $eye_color;
    
  public function __construct($color, $eye_color) {
    $this->legs = 4;
    $this->color = $color;
    $this->eye_color = $eye_color;
  }   
}

$rover = new Dog("black/white", "light blue");
var_dump($rover);

Constructor

<?php

class Dog {

  public $legs = 4;

}

$rover = new Dog();
echo $rover;

// returns a Fatal error.
<?php

class Dog {

  public $legs = 4;

  public function __toString() {
    return "Dogs have $this->legs legs.";
  }

}

$rover = new Dog();
echo $rover;

// outputs "Dogs have 4 legs."

__toString

Inheritance

class Shape 
{
    public $sides;
    public $fill_color;

    public function __construct($sides) {
        $this->sides = $sides;
    }

    public function get_angle() {
        return 360 / $this->sides;
    }
}

$hexagon = new Shape(6);

echo $hexagon->get_angle(); // --> 60
class Rectangle extends Shape
{
    public $sides = 4;
    public $length;
    public $width;

    public function __construct($width, $length){
        $this->length = length;
        $this->width = width;
    }
}
class Square extends Rectangle
{
    public function __construct($width) {
        $this->length = width;
        $this->width = width;
    }
}
$my_square = new Square();
echo $my_square->get_angle(); // --> 90

$my_hexagon = new Shape(6);
echo $my_square->get_angle(); // --> 120

Extends

class Rectangle extends Shape
{
    public $sides = 4;
    public $length;
    public $width;

    public function __construct($width, $length){
        $this->length = length;
        $this->width = width;
    }
}
class Shape 
{
    public $sides;
    public $fill_color;

    public function __construct($sides) {
        $this->sides = $sides;
    }

    public function get_angle() {
        return 360 / $this->sides;
    }
}

$hexagon = new Shape(6);

echo $hexagon->get_angle(); // --> 60

Override

class Rectangle extends Shape
{
    public $length;
    public $width;

    public function __construct($width, $length){
        parent::__construct(4);
        $this->length = length;
        $this->width = width;
    }
}
class Shape 
{
    public $sides;
    public $fill_color;

    public function __construct($sides) {
        $this->sides = $sides;
    }

    public function get_angle() {
        return 360 / $this->sides;
    }
}

$hexagon = new Shape(6);

echo $hexagon->get_angle(); // --> 60

Parent

Visibility

<?php
 
class Person
{
    public name;
    public weight = 0;    

    public function run($miles) {
        $this->weight -= $miles;
    }
}

Can be accessed anywhere, both within the class and externally.

<?php

$my_person = new Person();

// public property and method I can access
$my_person->weight = 100;
$my_person->run(5);

?>

Public

Can only be accessed within the class itself or in descendant classes (classes that extend the class containing the protected method).

Protected

<?php

class Shape 
{
    protected $sides;
    protected $fill_color;

    public function __construct($sides) {
        $this->sides = $sides;
    }

    public function get_angle() {
        return 360 / $this->sides;
    }

    public function get_sides() {
        return $this->sides;
    }
}
<?php

class Rectangle extends Shape{
    protected $length;
    protected $width;

    public function __construct($width, $length){
        $this->sides = 4; // <-- this is ok.
        $this->length = length;
        $this->width = width;
    }
}

// Accessing protected properties would be fatal
$rect = new Rectangle(3, 5);
echo $rect->sides; // --> fatal error
echo $rect->get_sides(); // --> this is ok.
?>

Accessible only from within the class that defines it. This means that even if a new class extends the class that defines a private property, that property or method will not be available at all within the child class.

Private

<?php
 
class Person
{
    protected $name;
    private $walks; // boolean  

    public function get_name() {
        return $this->name;
    }

    public function set_name($name) {
        $this->name = $name;
    }  

    public function get_walks() {
        return $this->walks;
    }

    public function set_walks($walks) {
        $this->walks = $walks;
    }
}
<?php

class Infant extends Person{

    public function __construct($name){
        $this->name = $name; // <-- Successfully sets the name to protected
        $this->walks = FALSE; // <-- NOT OK.
        $this->set_walks(FALSE); // <-- OK.
    }
}

$my_person = new Infant("Charlie");

// cannot access protected from outside the class
echo $my_person->name; // --> fatal error

// can run public methods outside the class
echo $my_person->get_walks(); // correctly outputs false

// but accessing private elements would be fatal
echo $my_person->walks; // --> fatal error

?>
<?php

class MyClass 
{
  public static function plusOne() {
      return "The count is " . ++self::$count . ".<br />";
  }
}

can be accessed without first instantiating the class

Note:

When accessing static properties, the dollar sign
($) comes after the scope resolution operator (::).

Static

<?php

do
{
  // Call plusOne without instantiating MyClass
  echo MyClass::plusOne();
} while ( MyClass::$count < 10 );

"In languages like c# and java, statics are a part of the language. You can do things like... Console.WriteLine() or int.TryParse(), String.Join() etc... these are all utilities"

Console.WriteLine();
int.TryParse();
String.Join();

 

Comments |When to Use Static Methods
FEB 20, 2015 / jrenton

http://verraes.net/2014/06/when-to-use-static-methods-in-php/#comment-1865595749

 

<?php
$time = Time::from("11:45");
<?php
$sum = Calculator::sum(1, 2);

 

When to Use Static Methods
JUN 14, 2014 / MATHIAS VERRAES

http://verraes.net/2014/06/when-to-use-static-methods-in-php/

 

Example use cases:

Static Examples

Comments

<?php

/**
 * Dear maintainer:
 * 
 * Once you are done trying to 'optimize' this routine,
 * and have realized what a terrible mistake that was,
 * please increment the following counter as a warning
 * to the next guy:
 * 
 * total_hours_wasted_here = 42
 */
<?php

// When I wrote this, only God and I understood what I was doing
// Now, God only knows
// drunk, fix later
long john; // silver
# TODO: Figure out what i'm doing here and comment accordingly
// Magic. Do not touch.
// John! If you’ll svn remove this once more,
// I’ll shut you, for God’s sake!
// That piece of code is not “something strange”!
// That is THE AUTH VALIDATION.

What are some of the funniest comments in source code? | Quora

Last asked: 22 Dec, 2013

http://www.quora.com/What-are-some-of-the-funniest-comments-in-source-code

 

  • Starts with a forward slash and two asterisks (/**)
  • Always associated with just one structural element in PHP
    • a file, class, interface, trait, function, constant, method, property or variable.
/**
  * Returns the name of this object.
  *
  * @return string
  */

Example

DocBlocks

Tags

  • Tags are a type of specialized information (meta-data)
  • Begins on a new line with an @ symbol
  • Arguments that provide additional context
<?php
 
/**
 * @author James Candan <james@codejourneymen.com>
 * @copyright 2015 Acme
 * @license http://www.php.net/license/3_01.txt PHP License 3.01

. . .

 * @param string $val a value required for the class
 * @return void
 */

Example

Example

<?php
 
/**
 * A simple class
 *
 * This is the long description for this class,
 * which can span as many lines as needed. It is
 * not required, whereas the short description is
 * necessary.
 *
 * It can also span multiple paragraphs if the
 * description merits that much verbiage.
 *
 * @author Jason Lengstorf <jason.lengstorf@ennuidesign.com>
 * @copyright 2010 Ennui Design
 * @license http://www.php.net/license/3_01.txt PHP License 3.01
 */
class SimpleClass
{
  /**
   * A public variable
   *
   * @var string stores data for the class
   */
  public $foo;
 
  /**
   * Sets $foo to a new value upon class instantiation
   *
   * @param string $val a value required for the class
   * @return void
   */
  public function __construct($val)
  {
      $this->foo = $val;
  }
 
  /**
   * Multiplies two integers
   *
   * Accepts a pair of integers and returns the
   * product of the two.
   *
   * @param int $bat a number to be multiplied
   * @param int $baz a number to be multiplied
   * @return int the product of the two parameters
   */
  public function bar($bat, $baz)
  {
      return $bat * $baz;
  }
}
 
?>

Auto-documentation

Tools such as phpDocumentor can be used to generate professional documentation directly from the source code of your PHP project. 

 

Namespacing in PHP | Tuts+
OCT 1, 2012 / 
ELIAS ZERROUQ

http://code.tutsplus.com/tutorials/namespacing-in-php--net-27203

 

Namespacing

What's a Namespace?

Namespacing is used to avoid conflicting definitions and introduce more flexibility and organization in your code base.

Defining a Namespace

A namespace definition

  • Must be first statement in a PHP file
  • Uses the "namespace" keyword
  • Must start with a letter or underscore, followed by any number of letters, numbers, or underscores
<?php 

/**
 * @file my_project.php
 */

namespace MyProject {
    // Regular PHP code goes here, anything goes!
    function run() 
    {
        echo 'Running from a namespace!';
    }
}
<?php

require 'my_project.php';

run();  // --> Fatal error: 
        // Call to undefined function run()


\MyProject\run(); // --> success!

Sub-namespaces

  • Follow a certain hierarchy, much like directories
  • You should store sub-namespaces in sub-directories
  • PHP uses the backslash as its namespace separator
  • You can have as many sub-namespaces as you want.
<?php 
/**
 * @file myproject/database/connection.php
 */

namespace MyProject\Database
 
class Connection {
    // Handling database connections
}
<?php 

/**
 * @file myproject/blog/auth/handler/social/twitter.php
 */

namespace MyProject\Blog\Auth\Handler\Social;
 
class Twitter {
    // Handles Twitter authentication
}

Comparison

<?php

/**
 * Let's create two person arrays, each 
 * containing a bit of info about the person.
 */

$person1 = array(
  'name' => 'Tom',
  'job' => 'Button-Pusher',
  'age' => 34
);
 
$person2 = array(
  'name' => 'John',
  'job' => 'Lever-Puller',
  'age' => 41
);

function changeJob($person, $new_job) {
  $person['job'] = $new_job; // Change job
  return $person;
}

function happyBirthday($person) {
  ++$person['age']; // Add 1 to age
  return $person;
}

// Tom got a promotion and had a birthday
$person1 = changeJob($person1, 'Box-Mover');
$person1 = happyBirthday($person1);
 
// John just had a birthday
$person2 = happyBirthday($person2);

<?php

/**
 * @file
 * Given two Linkedin contacts, we want to 
 * update their job title and age.
 */

class Person
{
  private $_name;
  private $_job;
  private $_age;

  public function __construct($name, $job, $age) {
    $this->_name = $name;
    $this->_job = $job;
    $this->_age = $age;
  }

  /**
   * Set person's job title.
   *
   * @param $newjob
   */
  public function changeJob($new_job) {
    $this->_job = $new_job;
  }

  /**
   * Increment person's age.
   */
  public function happyBirthday() {
    ++$this->_age;
  }

  // ...
  // getters and setters
}

 
/**
 * Let's create two person objects, each 
 * containing a bit of info about the person.
 */

$person1 = new Person("Tom", "Button-Pusher", 34);
$person2 = new Person("John", "Lever Puller", 41);


// Tom got a promotion and had a birthday
$person1->changeJob("Box-Mover");
$person1->happyBirthday();

// John just had a birthday
$person2->happyBirthday();

Procedural

OOP

Summary

  • Classes are blueprints
  • Objects are actual houses
  • Classes can have Parent - Child Inheritance via "extends"
  • Visibility
    • Public - can be seen from anywhere
    • Protected - can be seen from Class & Subclass
    • Private - can ONLY be seen from within Class
    • Static - Doesn't require instantiating (Utilities)
  • Namespaces - like directories

Questions?

OOP PHP for Drupalers

By James Candan

OOP PHP for Drupalers

  • 1,593