Php魔术函数学习与应用

    技术2022-05-19  21

    关键字: php, 魔术函数

    (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 附值 aninggo5,__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();从而输出打印 Test7,__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 serializationCODE: [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”的文件:

    < ? php /** * 测试__autoload方法 * */ class Test_autoload { public function __construct () { echo " Test_autoload. " ; } } ?>

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

    < ? php /** * 重写 __autoload方法 */ function __autoload ( $class ) { include $class . ' .php ' ; } $test = new Test_autoload () ; unset ( $test ) ; ?>

    最后结果为: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

     

     

    转自:http://zhengdl126.javaeye.com/blog/430959


    最新回复(0)