php的魔术方法

来源:互联网 发布:有机化学软件 编辑:程序博客网 时间:2024/04/27 22:53
2009-10-04 20:31

从php5以后的版本,类就可以使用魔术方法了。php规定以两个下划线(__)开头的方法都保留为魔术方法,所以建议大家函数名最好不用__开头,除非是为了重载已有的魔术方法。

目前php已有的魔术方法有__construct,__destruct,__call,__get,__set,__isset,__unset,__sleep,__wakeup,__toString,__set_state 和 __clone

__construct和__destruct是类的构造函数和析构函数,这个大家经常会用到,相信大家都很熟悉,这里就不多说了。

__sleep和__wakeup是序列化类的时候调用的。当序列化对象时,php将试图在序列动作之前调用该对象的成员函数__sleep(),当使用unserialize() 恢复对象时, 将调用__wakeup()。

 

__toString是对象被转为string时调用的

 

详情参考:http://mydraft.cn/posts/66

1.__construct()

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

class Test
{
function __construct()

{
echo "before";
}
}


$t = new Test();

输出是:

start
我们知道php5对象模型 和类名相同的函数是类的构造函数,那么如果我们同时定义构造函数和__construct()方法的话,php5会默认调用构造函数而不会调用__construct()函数,所以__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;

程序输出:
我被克隆了!

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