Tuesday, December 30, 2008

Magic Methods, __sleep , __wakeup, __toString, __set_state in PHP

Magic Methods

The function names __construct, __destruct (see Constructors and Destructors), __call, __callStatic, __get, __set, __isset, __unset (see Overloading), __sleep, __wakeup, __toString, __set_state and __clone are magical in PHP classes. You cannot have functions with these names in any of your classes unless you want the magic functionality associated with them



__sleep and __wakeup

serialize() checks if your class has a function with the magic name __sleep. If so, that function is executed prior to any serialization. It can clean up the object and is supposed to return an array with the names of all variables of that object that should be serialized. If the method doesn't return anything then NULL is serialized and E_NOTICE is issued.

The intended use of __sleep is to commit pending data or perform similar cleanup tasks. Also, the function is useful if you have very large objects which do not need to be saved completely.

Conversely, unserialize() checks for the presence of a function with the magic name __wakeup. If present, this function can reconstruct any resources that the object may have.

The intended use of __wakeup is to reestablish any database connections that may have been lost during serialization and perform other reinitialization tasks.


class Connection {
protected
$link;
private
$server, $username, $password, $db;

public function
__construct($server, $username, $password, $db)
{
$this->server = $server;
$this->username = $username;
$this->password = $password;
$this->db = $db;
$this->connect();
}

private function
connect()
{
$this->link = mysql_connect($this->server, $this->username, $this->password);
mysql_select_db($this->db, $this->link);
}

public function
__sleep()
{
return array(
'server', 'username', 'password', 'db');
}

public function
__wakeup()
{
$this->connect();
}
}
?>

__toString

The __toString method allows a class to decide how it will react when it is converted to a string.

Simple example

// Declare a simple class
class TestClass
{
public
$foo;

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

public function
__toString() {
return
$this->foo;
}
}

$class = new TestClass('Hello');
echo
$class;
?>

The above example will output:

Hello
It is worth noting that before PHP 5.2.0 the __toString method was only called when it was directly combined with echo() or print(). Since PHP 5.2.0, it is called in any string context (e.g. in printf() with %s modifier) but not in other types contexts (e.g. with %d modifier). Since PHP 5.2.0, converting objects without __toString method to string would cause E_RECOVERABLE_ERROR.

__set_state

This static method is called for classes exported by var_export() since PHP 5.1.0.

The only parameter of this method is an array containing exported properties in the form array('property' => value, ...).

Using __set_state (since PHP 5.1.0)


class A
{
public
$var1;
public
$var2;

public static function
__set_state($an_array) // As of PHP 5.1.0
{
$obj = new A;
$obj->var1 = $an_array['var1'];
$obj->var2 = $an_array['var2'];
return
$obj;
}
}

$a = new A;
$a->var1 = 5;
$a->var2 = 'foo';

eval(
'$b = ' . var_export($a, true) . ';'); // $b = A::__set_state(array(
// 'var1' => 5,
// 'var2' => 'foo',
// ));
var_dump($b);

?>

The above example will output:

object(A)#2 (2) {
["var1"]=>
int(5)
["var2"]=>
string(3) "foo"
}




Object cloning in PHP

Object cloning

Creating a copy of an object with fully replicated properties is not always the wanted behavior. A good example of the need for copy constructors, is if you have an object which represents a GTK window and the object holds the resource of this GTK window, when you create a duplicate you might want to create a new window with the same properties and have the new object hold the resource of the new window. Another example is if your object holds a reference to another object which it uses and when you replicate the parent object you want to create a new instance of this other object so that the replica has its own separate copy.

An object copy is created by using the clone keyword (which calls the object's __clone() method if possible). An object's __clone() method cannot be called directly.

$copy_of_object = clone $object;

When an object is cloned, PHP 5 will perform a shallow copy of all of the object's properties. Any properties that are references to other variables, will remain references. If a __clone() method is defined, then the newly created object's __clone() method will be called, to allow any necessary properties that need to be changed.

Cloning an object

class SubObject
{
static
$instances = 0;
public
$instance;

public function
__construct() {
$this->instance = ++self::$instances;
}

public function
__clone() {
$this->instance = ++self::$instances;
}
}

class
MyCloneable
{
public
$object1;
public
$object2;

function
__clone()
{
// Force a copy of this->object, otherwise
// it will point to same object.
$this->object1 = clone $this->object1;
}
}

$obj = new MyCloneable();

$obj->object1 = new SubObject();
$obj->object2 = new SubObject();

$obj2 = clone $obj;


print(
"Original Object:\n");
print_r($obj);

print(
"Cloned Object:\n");
print_r($obj2);

?>

The above example will output:

Original Object:
MyCloneable Object
(
[object1] => SubObject Object
(
[instance] => 1
)

[object2] => SubObject Object
(
[instance] => 2
)

)
Cloned Object:
MyCloneable Object
(
[object1] => SubObject Object
(
[instance] => 3
)

[object2] => SubObject Object
(
[instance] => 2
)

)


Overloading in PHP

Overloading

Overloading in PHP provides means to dynamically "create" members and methods. These dynamic entities are processed via magic methods one can establish in a class for various action types.

The overloading methods are invoked when interacting with members or methods that have not been declared or are not visible in the current scope. The rest of this section will use the terms "inaccessible members" and "inaccessible methods" to refer to this combination of declaration and visibility.

All overloading methods must be defined as public.

Note: None of the arguments of these magic methods can be passed by reference.

Note: PHP's interpretation of "overloading" is different than most object oriented languages. Overloading traditionally provides the ability to have multiple methods with the same name but different quantities and types of arguments.

ChangeLog

Version Description
5.1.0 Added __isset() and __unset().
5.3.0 Added __callStatic(). Added warning to enforce public visibility and non-static declaration.

Member overloading

void __set ( string $name , mixed $value )
mixed __get ( string $name )
bool __isset ( string $name )
void __unset ( string $name )

__set() is run when writing data to inaccessible members.

__get() is utilized for reading data from inaccessible members.

__isset() is triggered by calling isset() or empty() on inaccessible members.

__unset() is invoked when unset() is used on inaccessible members.

The $name argument is the name of the member being interacted with. The __set() method's $value argument specifies the value the $name'ed member should be set to.

Member overloading only works in object context. These magic methods will not be triggered in static context. Therefore these methods can not be declared static.

Example #1 overloading with __get, __set, __isset and __unset example

class MemberTest {
/** Location for overloaded data. */
private $data = array();

/** Overloading not used on declared members. */
public $declared = 1;

/** Overloading only used on this when accessed outside the class. */
private $hidden = 2;

public function
__set($name, $value) {
echo
"Setting '$name' to '$value'\n";
$this->data[$name] = $value;
}

public function
__get($name) {
echo
"Getting '$name'\n";
if (
array_key_exists($name, $this->data)) {
return
$this->data[$name];
}

$trace = debug_backtrace();
trigger_error(
'Undefined property via __get(): ' . $name .
' in ' . $trace[0]['file'] .
' on line ' . $trace[0]['line'],
E_USER_NOTICE);
return
null;
}

/** As of PHP 5.1.0 */
public function __isset($name) {
echo
"Is '$name' set?\n";
return isset(
$this->data[$name]);
}

/** As of PHP 5.1.0 */
public function __unset($name) {
echo
"Unsetting '$name'\n";
unset(
$this->data[$name]);
}

/** Not a magic method, just here for example. */
public function getHidden() {
return
$this->hidden;
}
}


echo
"
\n";


$obj = new MemberTest;

$obj->a = 1;
echo
$obj->a . "\n\n";

var_dump(isset($obj->a));
unset(
$obj->a);
var_dump(isset($obj->a));
echo
"\n";

echo
$obj->declared . "\n\n";

echo
"Let's experiment with the private property named 'hidden':\n";
echo
"Privates are visible inside the class, so __get() not used...\n";
echo
$obj->getHidden() . "\n";
echo
"Privates not visible outside of class, so __get() is used...\n";
echo
$obj->hidden . "\n";
?>

The above example will output:

Setting 'a' to '1'

Getting 'a'
1

Is 'a' set?
bool(true)
Unsetting 'a'
Is 'a' set?
bool(false)

1

Let's experiment with the private property named 'hidden':
Privates are visible inside the class, so __get() not used...
2
Privates not visible outside of class, so __get() is used...
Getting 'hidden'


Notice: Undefined property via __get(): hidden in on line 70 in on line 29

Method overloading

mixed __call ( string $name , array $arguments )
mixed __callStatic ( string $name , array $arguments )

__call() is triggered when invoking inaccessible methods in an object context.

__callStatic() is triggered when invoking inaccessible methods in a static context.

The $name argument is the name of the method being called. The $arguments argument is an enumerated array containing the parameters passed to the $name'ed method.

Example #2 overloading instantiated methods with __call and ___callStatic

class MethodTest {
public function
__call($name, $arguments) {
// Note: value of $name is case sensitive.
echo "Calling object method '$name' "
. implode(', ', $arguments). "\n";
}

/** As of PHP 5.3.0 */
public static function __callStatic($name, $arguments) {
// Note: value of $name is case sensitive.
echo "Calling static method '$name' "
. implode(', ', $arguments). "\n";
}
}

$obj = new MethodTest;
$obj->runTest('in object context');

MethodTest::runTest('in static context'); // As of PHP 5.3.0
?>

The above example will output:

Calling object method 'runTest' in object context

Calling static method 'runTest' in static context

Wednesday, December 24, 2008

Free PHP books links

http://rapidshare.com/files/128636166/04_-_PHP_5_and_MySQL_Bible.rar

http://rapidshare.com/files/128637790/09_-_PHP_Professional_Projects.rar

http://rapidshare.com/files/144017451/Ajax_in_Action.pdf

http://rapidshare.com/files/143551939/Beginning_PHP_and_MySQL_5_From_Novice_to_Professional.pdf

http://rapidshare.com/files/144018301/Beginning_XML_With_DOM_And_Ajax_-_From_Novice_To_Professional.pdf

http://rapidshare.com/files/128391727/Building_Online_Communities_With_Drupal__Phpbb_And_Wordpress.pdf

http://rapidshare.com/files/144017621/Dreamweaver_CS3_with_CSS_Ajax_and_PHP.part1.rar
http://rapidshare.com/files/144017789/Dreamweaver_CS3_with_CSS_Ajax_and_PHP.part2.rar

http://rapidshare.com/files/133735694/Foundations_of_PEAR_Rapid_PHP_Development.pdf

http://rapidshare.com/files/128392357/Learning_Drupal_6_Module_Development_2008.pdf

http://rapidshare.com/files/143548475/PHP5_CMS_Framework_Development.pdf

http://rapidshare.com/files/143550423/PHP_and_MySQL_-__Create-Modify-Reuse.pdf

http://rapidshare.com/files/143550793/PHP_Objects__Patterns__and_Practice.pdf

http://rapidshare.com/files/144016785/Understanding_AJAX_-_Using_JavaScript_To_Create_Rich_Internet_Applications.chm

http://rapidshare.com/files/128392179/Pro_Drupal_Development_2007.pdf

http://rapidshare.com/files/143549330/Secure_PHP_Development_Building_50_Practical_Applications.pdf

http://rapidshare.com/files/143549916/Teach_Yourself_Php__Mysql_And_Apache_All_In_One.chm

http://rapidshare.com/files/144017964/Practical_JavaScript__DOM_Scripting__and_Ajax_Projects.part1.rar
http://rapidshare.com/files/144018096/Practical_JavaScript__DOM_Scripting__and_Ajax_Projects.part2.rar

http://rapidshare.com/files/143549629/Practical_Web_2.0_Applications_with_PHP.pdf

http://rapidshare.com/files/133735694/Foundations_of_PEAR_Rapid_PHP_Development.pdf


http://rapidshare.com/files/128637790/09_-_PHP_Professional_Projects.rar

http://rapidshare.com/files/128636166/04_-_PHP_5_and_MySQL_Bible.rar