欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Php魔术函数学习与应用 __construct() __destruct() __get()等

程序员文章站 2022-06-05 18:45:37
...

(1)初识魔术方法
Php5.0发布以来为我们提供了很多面向对象 的特性,尤其是为我们提供了好多易用的魔术方法,这些魔术方法可以让我们简化我们的编码,更好的设计我们的系统。今天我们就来认识下php5.0给我们提供的魔术方法。

1,__construct() 当实例化一个对象的时候,这个对象的这个方法首先被调用。


class Test
{
function __construct()

{
echo "before";
}
}


$t = new Test();


输出是:

start

 

 

<?
class Test
{
    function Test()
{
echo "end2";
}
function __construct()
{
echo "end";
}

}
$t = new Test();

//输出end
?>
我们知道php5对象模型 和类名相同函数 是类的构造函数 ,那么如果我们同时定义构造函数 和__construct()方法的话,php5会默认调用构造函数 而不会调用同类名函数 ,所以__construct()作为类的默认的构造函数


2,__destruct() 当删除一个对象或对象操作终止的时候,调用该方法。


class Test
{
function __destruct()
{
echo "end";
}
}
$t = new Test();
将会输出
end


我们就可以在对象操作结束的时候进行释放资源之类的操作

3,__get() 当试图读取一个并不存在的属性的时候被调用。
如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数 实现类似java中反射的各种操作。


class Test
{
public function __get($key)
{
echo $key . " 不存在";
}
}


$t = new Test();
echo $t->name;

就会输出:
name 不存在


4,__set() 当试图向一个并不存在的属性写入值的时候被调用。


class Test
{
public function __set($key,$value)
{
echo '对'.$key . "附值".$value;
}
}


$t = new Test();
$t->name = "aninggo";

就会输出:
对 name 附值 aninggo


5,__call() 当试图调用一个对象并不存在的方法时,调用该方法。
class Test
{
public function __call($Key, $Args)
{
echo "您要调用的 {$Key} 方法不存在。你传入的参数是:" . print_r($Args, true);
}
}

$t = new Test();
$t->getName(aning,go);



程序将会输出:
您要调用的 getName 方法不存在。参数是:Array
(
[0] => aning
[1] => go
)

6,__toString() 当打印一个对象的时候被调用
这个方法类似于java的toString方法,当我们直接打印对象的时候回调用这个函数
class Test
{
public function __toString()
{
return "打印 Test";
}
}


$t = new Test();

echo $t;




运行echo $t;的时候,就会调用$t->__toString();从而输出
打印 Test

7,__clone() 当对象被克隆时,被调用
class Test
{

public function __clone()
{
echo "我被复制了!";
}
}

$t = new Test();
$t1 = clone $t;

程序输出:
我被克隆了!

 

 

__sleep 和 __wakeup

 

 

串行化serialize可以把变量包括对象,转化成连续bytes数据. 你可以将串行化后的变量存在一个文件里或在网络上传输. 然后再反串行化还原为原来的数据. 你在反串行化类的对象之前定义的类,PHP可以成功地存储其对象的属性和方法. 有时你可能需要一个对象在反串行化后立即执行. 为了这样的目的,PHP会自动寻找__sleep和__wakeup方法.

  当一个对象被串行化,PHP会调用__sleep方法(如果存在的话). 在反串行化一个对象后,PHP 会调用__wakeup方法. 这两个方法都不接受参数. __sleep方法必须返回一个数组,包含需要串行化的属性. PHP会抛弃其它属性的值. 如果没有__sleep方法,PHP将保存所有属性.

  例子6.16显示了如何用__sleep和__wakeup方法来串行化一个对象. Id属性是一个不打算保留在对象中的临时属性. __sleep方法保证在串行化的对象中不包含id属性. 当反串行化一个User对象,__wakeup方法建立id属性的新值. 这个例子被设计成自我保持. 在实际开发中,你可能发现包含资源(如图像或数据流)的对象需要这些方法

Object serialization
CODE: [Copy to clipboard]
--------------------------------------------------------------------------------

<?php

class User
{
public $name;
public $id;

function __construct()
{
//give user a unique ID 赋予一个不同的ID
$this->id = uniqid();
}

function __sleep()
{
//do not serialize this->id 不串行化id
return(array("name"));
}

function __wakeup()
{
//give user a unique ID
$this->id = uniqid();
}
}

//create object 建立一个对象
$u = new User;
$u->name = "Leon";

//serialize it 串行化 注意不串行化id属性,id的值被抛弃
$s = serialize($u);

//unserialize it 反串行化 id被重新赋值
$u2 = unserialize($s);

//$u and $u2 have different IDs $u和$u2有不同的ID
print_r($u);
print_r($u2);
?>

 

 

 

 

__set_state and __invoke

 

测试代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<?php

 
class
 A {

    public
 static function
 __set_state(
$args
)
 {

        $obj
 =
 new
 A(
)
;

        foreach
(
$args
 as
 $k
=>
$v
)
 {

		     $obj
->
$k
 =
 $v
;

		}

        return
 $obj
;

    }

}

 
$a
 =
 new
 A;

$a
->
name
 =
 'cluries'
;

$a
->
sex
 =
 'female'
;

 
eval
(
'$b = '
 .
 var_export
(
$a
,
 true
)
 .
 ';'
)
;

 
print_r
(
$b
)
;

?>

程序输出

object(A)#2 (2) {
         ["name"]=>  string(7) "cluries"
         ["sex"]=>  string(6) "female"
}

得出以下结论,__set_state作用是用来复制一个对象,并且可以在__set_state中定义在复制对象的时候对复制得到的对象进行一些 改变。和__clone不同的是__set_state可以接受参数,__set_state使用起来更加强大!虽然个人觉得这个东西不是很好用= =!

 

 

然后再说下__invoke:
手册上有个非常显眼的:Note: This feature is available since PHP 5.3.0. 

The __invoke method is called when a script tries to call an object as a function.

__invoke方法将会在代码试图把对象当作函数来使用时候调用?有点稀奇,这个功能有什么用处呢?
然后看下提供的例子:

1
2
3
4
5
6
7
8
9
10
<?php

class
 CallableClass {

    function
 __invoke(
$x
)
 {

        var_dump
(
$x
)
;

    }

}

$obj
 =
 new
 CallableClass;

$obj
(
5
)
;

var_dump
(
is_callable
(
$obj
)
)
;

?>

程序输出:

int(5)
bool(true)

还真是把对象当函数使用…

 

 

 

 

__autoload

 

 

PHP5中有一方法: __autoload() , 简单的说就是类的自动加载;

当你尝试使用一个PHP没有组织到的类, 它会寻找一个__autoload的全局函数. 如果存在这个函数,PHP会用一个参数来调用它,参数即类的名称。

那么简单测试一下。


首先建一个名为”Test_autoload.php”的文件:

  1. < ? php
  2. /**
  3. * 测试__autoload方法
  4. *
  5. */
  6. class Test_autoload {
  7. public function __construct () {
  8. echo " Test_autoload. " ;
  9. }
  10. }
  11. ?>

注意类名哦 , 然后随便建个文件重写 __autoload() 方法,这里假设是”test.php”;

  1. < ? php
  2. /**
  3. * 重写 __autoload方法
  4. */
  5. function __autoload ( $class ) {
  6. include $class . ' .php ' ;
  7. }

  8. $test = new Test_autoload () ;
  9. unset ( $test ) ;
  10. ?>

最后结果为:Test_autoload.

 


------------------------------------------------------------------------------------
8.顺便介绍下php5中提供的几个非常COOl的实验性函数
(1)。runkit_method_rename
    这个函数 可以动态的改变我们所调用的函数 名字


class Test
{

function foo() {
        return "foo! ";
    }

}

runkit_method_rename(
    'Test', //类名
    'foo',//实际调用的函数
    'bar'//显示调用的函数
);

echo Test::bar();

程序将输出
 
foo!



(2) runkit_method_add

这个函数 可以动态的向类中添加函数


class Test
{

function foo() {
        return "foo! ";
    }

}

runkit_method_add(
    Test, //类名
    'add', //新函数
    '$num1, $num2',//传入参数
    'return $num1 + $num2;',//执行的代码
    RUNKIT_ACC_PUBLIC
);

// 调用
echo $e->add(12, 4);



(3)runkit_method_copy
可以把A类中的函数 拷贝到类B中并对函数 重命名


class Foo {
    function example() {
        return "foo! ";
    }
}

class Bar {
    //空类
}

//执行拷贝
runkit_method_copy('Bar', 'baz', 'Foo', 'example');

//执行拷贝后的函数
echo Bar::baz();



(4) runkit_method_redefine
动态的修改函数 的返回值
这个函数 可以让我们轻松的实现对类的MOCK测试!是不是很COOL呢


class Example {
    function foo() {
        return "foo! ";
    }
}

//创建一个测试对象
$e = new Example();

// 在测试对象之前输出
echo "Before: " . $e->foo();

// 修改返回值
runkit_method_redefine(
    'Example',
    'foo',
    '',
    'return "bar! ";',
    RUNKIT_ACC_PUBLIC
);

// 执行输出
echo "After: " . $e->foo();



(5)runkit_method_remove
这个函数 就很简单了,看名字 就能看出来了,动态的从类中移除函数
class Test {
    function foo() {
        return "foo! ";
    }
   
    function bar() {
        return "bar! ";
    }
}

// 移除foo函数
runkit_method_remove(
    'Test',
    'foo'
);

echo implode(' ', get_class_methods('Test'));

程序输出
bar