PHP魔术函数学习实例
1,__construct() 当实例化一个对象的时候,这个对象的这个方法首先被调用。
(1)初识魔术方法
| 1 2 3 4 5 6 7 8 9 | classTest { function__construct() { echo“before”; } } $t=newTest(); |
输出是:
start
我们知道php5对象模型 和类名相同的函数是类的构造函数,那么如果我们同时定义构造函数和__construct()方法的话,php5会默认调用构造函数而不会调用__construct()函数,所以__construct()作为类的默认的构造函数
我们知道php5对象模型 和类名相同的函数是类的构造函数,那么如果我们同时定义构造函数和__construct()方法的话,php5会默认调用构造函数而不会调用__construct()函数,所以__construct()作为类的默认的构造函数
2,__destruct() 当删除一个对象或对象操作终止的时候,调用该方法。
| 1 2 3 4 5 6 7 8 | classTest { function__destruct() { echo“end”; } } $t=newTest(); |
将会输出
end
我们就可以在对象操作结束的时候进行释放资源之类的操作
3,__get() 当试图读取一个并不存在的属性的时候被调用。
如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数实现类似java中反射的各种操作。
如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数实现类似java中反射的各种操作。
| 1 2 3 4 5 6 7 8 9 10 | classTest { publicfunction__get($key) { echo$key. ” 不存在”; } } $t=newTest(); echo$t->name; |
就会输出:
name 不存在
name 不存在
4,__set() 当试图向一个并不存在的属性写入值的时候被调用。
| 1 2 3 4 5 6 7 8 9 10 | classTest { publicfunction__set($key,$value) { echo‘对’.$key. “附值”.$value; } } $t=newTest(); $t->name = “aninggo”; |
就会输出:
对 name 附值 aninggo
对 name 附值 aninggo
5,__call() 当试图调用一个对象并不存在的方法时,调用该方法。
| 1 2 3 4 5 6 7 8 9 10 | classTest { publicfunction__call($Key,$Args) { echo“您要调用的 {$Key} 方法不存在。你传入的参数是:” . print_r($Args, true); } } $t=newTest(); $t->getName(aning,go); |
程序将会输出:
| 1 2 3 4 5 | 您要调用的 getName 方法不存在。参数是:Array ( [0] => aning [1] => go ) |
6,__toString() 当打印一个对象的时候被调用
这个方法类似于java的toString方法,当我们直接打印对象的时候回调用这个函数
这个方法类似于java的toString方法,当我们直接打印对象的时候回调用这个函数
| 1 2 3 4 5 6 7 8 9 10 11 | classTest { publicfunction__toString() { return“打印 Test”; } } $t=newTest(); echo$t; |
运行echo $t;的时候,就会调用$t->__toString();从而输出
打印 Test
打印 Test
7,__clone() 当对象被克隆时,被调用
| 1 2 3 4 5 6 7 8 9 10 11 | classTest { publicfunction__clone() { echo“我被复制了!”; } } $t=newTest(); $t1=clone$t; |
程序输出:
我被克隆了!
我被克隆了!
8.顺便介绍下php5中提供的几个非常COOl的实验性函数
(1)runkit_method_rename
这个函数可以动态的改变我们所调用的函数的名字。
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | classTest { functionfoo() { return“foo! “; } } runkit_method_rename( ‘Test’,//类名 ‘foo’,//实际调用的函数 ‘bar’//显示调用的函数 ); echoTest::bar(); |
程序将输出
foo!
(2) runkit_method_add
这个函数可以动态的向类中添加函数
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | classTest { functionfoo() { 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中并对函数重命名
可以把A类中的函数拷贝到类B中并对函数重命名
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | classFoo { functionexample() { return“foo! “; } } classBar { //空类 } //执行拷贝 runkit_method_copy(’Bar’, ‘baz’, ‘Foo’, ‘example’); //执行拷贝后的函数 echoBar::baz(); |
(4) runkit_method_redefine
动态的修改函数的返回值
这个函数可以让我们轻松的实现对类的MOCK测试!是不是很COOL呢
动态的修改函数的返回值
这个函数可以让我们轻松的实现对类的MOCK测试!是不是很COOL呢
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | classExample { functionfoo() { return“foo! “; } } //创建一个测试对象 $e=newExample(); // 在测试对象之前输出 echo“Before: ” .$e->foo(); // 修改返回值 runkit_method_redefine( ‘Example’, ‘foo’, ”, ‘return“bar! “;’, RUNKIT_ACC_PUBLIC ); // 执行输出 echo“After: ” .$e->foo(); |
(5)runkit_method_remove
这个函数就很简单了,看名字就能看出来了,动态的从类中移除函数
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | classTest { functionfoo() { return“foo! “; } functionbar() { return“bar! “; } } // 移除foo函数 runkit_method_remove( ‘Test’, ‘foo’ ); echoimplode(’ ‘, get_class_methods(’Test’)); |
程序输出
bar
bar