Tag: OOP

OOP中__clone(),__toString(),__autoload(),__destruct的使用

没有评论

2010 年 12 月 10 日 at 下午 3:49分类:PHP | WEB开发

有的时候我们需要在一个项目里面,使用两个或多个一样的对象,如果你使用“new”关键字重新创建对象的话,再赋值上相同的属性,这样做比较烦琐而且也容易出错,所以要根据一个对象完全克隆出一个一模一样的对象,是非常有必要的,而且克隆以后,两个对象互不干扰。
在PHP5中我们使用”clone”这个关键字克隆对象;

<? 
    class Person 
    { 
        //下面是人的成员属性 
        var $name;  //人的名子 
        var $sex;    //人的性别 
        var $age;    //人的年龄 
        //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值 
        function __construct($name="", $sex="", $age="") 
        { 
            $this->name=$name; 
            $this->sex=$sex; 
            $this->age=$age; 
        } 
        //这个人可以说话的方法, 说出自己的属性 
        function say()   
        { 
            echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."<br>"; 
        } 
    } 
    $p1=new Person("张三", "男", 20); 
    //使用“clone”克隆新对象p2,和p1对象具有相同的属性和方法。 
    $p2=clone $p1; 
    $p2->say(); 
?> 

PHP5定义了一个特殊的方法名“__clone()”方法,是在对象克隆时自动调用的方法,用“__clone()”方法将建立一个与原对象拥有相同属性和方法的对象,如果想在克隆后改变原对象的内容,需要在__clone()中重写原本的属性和方法, ”__clone()”方法可以没有参数,它自动包含$this和$that两个指针,$this指向复本,而$that指向原本;

<?
class Person 
{ 
    //下面是人的成员属性 
    var $name;  //人的名子 
    var $sex;    //人的性别 
    var $age;    //人的年龄 
    //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值 
    function __construct($name="", $sex="", $age="") 
    { 
        $this->name=$name; 
        $this->sex=$sex; 
        $this->age=$age; 
    } 
    //这个人可以说话的方法, 说出自己的属性 
    function say()   
    { 
        echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."<br>"; 
    } 
    //对象克隆时自动调用的方法, 如果想在克隆后改变原对象的内容,需要在__clone()中重写原本的属性和方法 
    function __clone() 
    { 
        //$this指的复本p2, 而$that是指向原本p1,这样就在本方法里,改变了复本的属性。  
        $this->name="我是假的$that->name"; 
        $this->age=30; 
    } 
} 
$p1=new Person("张三", "男", 20); 
$p2=clone $p1; 
$p1->say(); 
$p2->say(); 
?> 

上例输出:

我的名子叫:张三 性别:男 我的年龄是:20
我的名子叫:我是假的张三 性别:男 我的年龄是:30  

__toString()”方法也是一样自动被调用的,是在直接输出对象引用时自动调用的, 前面我们讲过对象引用是一个指针,比如说:“$p=new Person()“中,$p就是一个引用,我们不能使用echo 直接输出$p, 这样会输出”Catchable fatal error: Object of class Person could not be converted to string“这样的错误,如果你在类里面定义了“__toString()”方法,在直接输出对象引用的时候,就不会产生错误,而是自动调用了”__toString()”方法, 输出“__toString()”方法中返回的字符,所以“__toString()”方法一定要有个返回值(return 语句).

<?php
    // Declare a simple class
    class TestClass
    {
        public $foo;
        public function __construct($foo) {
        $this->foo = $foo;
        }
       //定义一个__toString方法,返加一个成员属性$foo
        public function __toString() {
        return $this->foo;
        }
    }
    $class = new TestClass('Hello'); 
    //直接输出对象
    echo $class;
?> 

上例输出:Hello
__autoload的用法相当的简单,我们一般在实例化一个类时,都要include一个类文件,那么要是我们在所有实例化之前先定义这么一个方法的的话,
那么就不用每次都include了 代码如下:

function __autoload($className){
        include($className.".php");            //先后加载MyPc.php和FBB.php这两个类文件
}
$pp = new MyPc();
$bb = new FBB();

__destruct()函数是在该类执行所有的操作动作之后被自动执行的,用来销毁该类的对象的!
只要我们在类中定义了,就不用管他了,在所有结束后,该类会自动执行!

OOP中__set() __get() __isset() __unset()四个方法的应用

没有评论

2010 年 12 月 10 日 at 下午 3:27分类:PHP | WEB开发

一般来说,总是把类的属性定义为private,这更符合现实的逻辑。但是, 对属性的读取和赋值操作是非常频繁的,因此在PHP5中,预定义了两个函数”__get()”和”__set()”来获取和赋值其属性,以及检查属性的”__isset()”和删除属性的方法”__unset()”。
”__set()”和“__get()”这两个方法不是默认存在的, 而是我们手工添加到类里面去的,像构造方法(__construct())一样, 类里面添加了才会存在,可以按下面的方式来添加这两个方法,当然也可以按个人的风格来添加:

//__get()方法用来获取私有属性
function __get($property_name)
{
    if(isset($this->$property_name)) {
        return($this->$property_name);
    }else {
        return(NULL);
    }
} 
//__set()方法用来设置私有属性
function __set($property_name, $value)
{
    $this->$property_name = $value;
}

__get()方法:这个方法用来获取私有成员属性值的,有一个参数, 参数传入你要获取的成员属性的名称,返回获取的属性值, 这个方法不用我们手工的去调用, 是在直接获取私有属性的时候自动调用的。因为私有属性已经被封装上了,是不能直接获取值的(比如:”echo $p1->name” 这样直接获取是错误的),但是如果你在类里面加上了这个方法,在使用”echo $p1->name” 这样的语句直接获取值的时候就会自动调用__get($property_name)方法,将属性name传给参数$property_name,通过这个方法的内部执行,返回我们传入的私有属性的值。
__set()方法:这个方法用来为私有成员属性设置值的,有两个参数,第一个参数为你要为设置值的属性名,第二个参数是要给属性设置的值,没有返回值。这个方法同样不用我们手工去调用,是在直接设置私有属性值的时候自动调用的,同样属性私有的已经被封装上了, 如果没有__set()这个方法,是不允许的, 比如:”$this->name=’zhangsan’ , 这样会出错,但是如果你在类里面加上了__set($property_name, $value)这个方法,在直接给私有属性赋值的时候,就会自动调用它,把属性比如name传给$property_name, 把要赋的值”zhangsan”传给$value,通过这个方法的执行,达到赋值的目的, 为了不传入非法的值, 还可以在这个方法给做一下判断。代码如下:

<?php
    class Person
    {
    //下面是人的成员属性, 都是封装的私有成员
    private $name; //人的名子
    private $sex; //人的性别
    private $age; //人的年龄

    //__get()方法用来获取私有属性
    function __get($property_name)
    {
        echo "在直接获取私有属性值的时候,自动调用了这个__get()方法";
        if(isset($this->$property_name)) {
            return($this->$property_name);
        }else {
            return(NULL);
        }
    } 
    //__set()方法用来设置私有属性
    function __set($property_name, $value)
    {
        echo "在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值";
        $this->$property_name = $value;
    } 
    }
    $p1=new Person();
    //直接为私有属性赋值的操作, 会自动调用__set()方法进行赋值
    $p1->name="张三";
    $p1->sex="男";
    $p1->age=20;
    //直接获取私有属性的值, 会自动调用__get()方法,返回成员属性的值
    echo "姓名:".$p1->name;
    echo "性别:".$p1->sex;
    echo "年龄:".$p1->age;
?>

程序执行结果:(省略)
在直接设置私有属性值的时候, 自动调用了这个__set()方法为私有属性赋值,在直接获取私有属性值的时候,自动调用了这个__get()方法姓名:张三在直接获取私有属性值的时候,自动调用了这个__get()方法获取性别:男在直接获取私有属性值的时候,自动调用了这个__get()方法年龄:20
以上代码如果不加上__get()和__set()方法,程序就会出错,因为不能在类的外部操作私有成员,而上面的代码是通过自动调用__get()和__set()方法来帮助我们直接存取封装的私有成员的。
__isset() 方法:检查属性是否存在的方法。
__unset()方法:删除指定属性的方法。
最后补充说明:
  __set(), __get() 是专门为类的私有属性设立的,对于类的公开(public)属性,外面是可以直接访问与设置的(如:$p1->name)!!!

此外还要补充一下:在OOP中我们还用到的函数有
__call()函数:这个函数用来处理错误调用
程序开发中,如果在使用对象调用对象内部方法时候,调用的这个方法不存在那么程序就会出错,然后程序退出不能继续执行。那么可不可以在程序调用对象内部不存在的方法时,提示我们调用的方法及使用的参数不存在,但程序还可以继续执行,这个时候我们就要使用在调用不存在的方法时自动调用的方法”__call()”.

<?php 
    //这是一个测试的类,里面没有属性和方法 
    class Test 
    { 
    } 
    //产生一个Test类的对象 
    $test=new Test(); 
    //调用对象里不存在的方法
    $test->demo("one", "two", "three"); 
    //程序不会执行到这里 
    echo "this is a test<br>"; 
?> 

上例出现如下错误,程序通出不能继续执行;

Fatal error: Call to undefined method Test::demo() 

下面我们加上“__call()”方法,这个方法有2个参数,第一个参数为调用不存在的方法过程中,自动调用__call()方法时,把这个不存在的方法的方法名传给第一个参数,第二个参数则是把这个方法的多个参数以数组的形式传进来。

<?php 
    //这是一个测试的类,里面没有属性和方法 
    class Test 
    { 
        //调用不存的方法时自动调用的方法,第一个参数为方法名,第二个参数是数组参数 
        function __call($function_name, $args) 
        {  
            print "你所调用的函数:$function_name(参数:";  
            print_r($args);  
            echo "不存在!<br>\n"; 
        } 
    } 
    //产生一个Test类的对象 
    $test=new Test(); 
    //调用对象里不存在的方法 
    $test->demo("one", "two", "three"); 
    //程序不会退出可以执行到这里 
    echo "this is a test<br>"; 
?> 

上例输出结果为:
你所调用的函数: demo(参数:Array ( [0] => one [1] => two [2] => three ) )不存在!
上面讲的就是一个简单的测试例子!