Chinaunix首页 | 论坛 | 博客
  • 博客访问: 342331
  • 博文数量: 121
  • 博客积分: 2771
  • 博客等级: 少校
  • 技术积分: 705
  • 用 户 组: 普通用户
  • 注册时间: 2010-12-01 12:44
文章分类

全部博文(121)

文章存档

2011年(121)

分类: Python/Ruby

2011-04-07 17:30:09

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

1,__construct() 当实例化一个对象的时候,这个对象的这个方法首先被调用。//面向对象基础,定义一个类使用一个对象 //定义一个类 这里class Class 是没有区别的 Class Person{ //定义属性 //Public(公开): 可以自由的在类的内部外部读取、修改。 //Private(私有): 只能在这个当前类的内部读取、修改。 //Protected(受保护):能够在这个类和类的子类中读取和修改。 private $name = "noName"; private $age = 20; function __construct() { echo 'ok
'; } function __get($key) { echo $key.':不可以这样拿我的名字
'; } function __set($key,$value) { echo '
'.$key.':不可以直接赋值'.$value.'
'; } function __destruct() { echo '
end'; } public function __toString() { //$this 指当前对象。 //在方法内部通过 $this-> 符号调用同一对象的属性。 return "我的名称:".$this->name."   我的年龄:".$this->age; } } $p = new Person(); var_dump($p); echo "
"; print_r($p); echo "
"; echo $p;//会自动调用类中的__toString()方法,所以如果这样做就必须在类中声明一个__toString() echo "
"; echo "我的名字是:".$p->name; echo "
"; echo "我的年龄是:".$p->age; //这里我们改变对象的属性,由于属性是public所以允许改变 $p->name = "Tom"; /* $p->age = 25; echo "
"; echo "我的名字是:".$p->name; echo "
"; echo "我的年龄是:".$p->age; */


我们知道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

 

------------------------------------------------------------

 

PHP把所有以__(两个下划线)开头的类方法当成魔术方法。所以你定义自己的类方法时,不要以 __为前缀。
 

1、__construct()

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

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

 

2、__destruct()

析构函数会在到某个对象的所有引用都被删除或者当对象被显式销毁时执行。

 

3、__get(string $name)

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

 

4、__set(string $name, mixed $value)

给未定义的变量赋值时将被调用

 

5、__call(string $name, array $arguments)

当调用一个不可访问方法(如未定义,或者不可见)时,__call() 会被调用。

__callStatic(string $name, array $arguments)

当在静态方法中调用一个不可访问方法(如未定义,或者不可见)时,__callStatic() 会被调用。

 

6、__toString()

当打印一个对象的时候被调用,这个方法类似于java的toString方法,当我们直接打印对象的时候回调这个函数。

 

7、__clone()

当对象被克隆时调用。

 

8、__sleep()

serialize()函数会检查是否存在一个魔术方法 __sleep.如果存在,__sleep()方法会先被调用, 然后才执行序列化操作。这个功能可以用于清理对象,并返回一个包含对象中所有变量名称的数组。如果该方法不返回任何内容,则NULL被序列化,导致 一个E_NOTICE错误。__sleep方法常用于提交未提交的数据,或类似的操作。同时,如果你有一些很大的对象,不需要保存,这个功能就很好用。

 

9、__wakeup()

与__sleep() 相反,unserialize()会检查是否存在一个__wakeup方法。如果存在,则会先调用__wakeup方法,预先准备对象数据。 __wakeup经常用在反序列化操作中,例如重新建立数据库连接,或执行其它初始化操作。

 

10、__isset()

当对未定义的变量调用isset() 或 empty()时,__isset() 会被调用。

 

11、__unset()

unset一个对象的属性时被调用。如:unset($c->name)。

 

12、__set_state()

调用var_export时,被调用。用__set_state的返回值做为var_export的返回值。

 

13、__autoload()

实例化一个对象时,如果对应的类不存在,则该方法被调用。简单的说就是类的自动加载;当你尝试使用一个PHP没有组织到的类, 它会寻找一个__autoload的全局函数. 如果存在这个函数,PHP会用一个参数来调用它,参数即类的名称。

 

14、__invoke()

当尝试以调用函数的方式调用一个对象时,__invoke 方法会被自动调用。

 

魔术常量:

__LINE__ 文件中的当前行号。 

__FILE__ 文件的完整路径和文件名。如果用在被包含文件中,则返回被包含的文件名。自 PHP 4.0.2 起,__FILE__ 总是包含一个绝对路径(如果是符号连接,则是解析后的绝对路径),而在此之前的版本有时会包含一个相对路径。 

__DIR__ 文件所在的目录。如果用在被包括文件中,则返回被包括的文件所在的目录。它等价于 dirname(__FILE__)。除非是根目录,否则目录中名不包括末尾的斜杠。(PHP 5.3.0中新增) = 

__FUNCTION__ 函数名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该函数被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。 

__CLASS__ 类的名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该类被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。 

__METHOD__ 类的方法名(PHP 5.0.0 新加)。返回该方法被定义时的名字(区分大小写)。 

__NAMESPACE__ 当前命名空间的名称(大小写敏感)。这个常量是在编译时定义的(PHP 5.3.0 新增)

阅读(1411) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~