Rumah >pembangunan bahagian belakang >tutorial php >php面向对象抽象方法和抽象类 __call 克隆对象详细教程_PHP教程

php面向对象抽象方法和抽象类 __call 克隆对象详细教程_PHP教程

WBOY
WBOYasal
2016-07-13 17:13:111046semak imbas

本文章介绍了关于php中面向对象的抽象方法和抽象类 __call 克隆对象用法,有需要发解的朋友可以学学。

抽象方法和抽象类


在OOP语言中,一个类可以有一个或多个子类,而每个类都有至少一个公有方法做为外部代码访问其的接口。而抽象方法就是为了方便继承而引入的,我们先来看一下抽象类和抽象方法的定义再说明它的用途。

什么是抽象方法?我们在类里面定义的没有方法体的方法就是抽象方法,所谓的没有方法体指的是,在方法声明的时候没有大括号以及其中的内容,而是直接在声明时在方法名后加上分号结束,另外在声明抽象方法时还要加一个关键字“abstract”来修饰;例如:

 代码如下 复制代码

abstract function fun1();
abstract function fun2();

上例是就是“abstract”修饰的没有方法体的抽象方法“fun1()”和“fun2()”,不要忘记抽象方法后面还要有一个分号;那么什么是抽象类呢?只要一个类里面有一个方法是抽象方法,那么这个类就要定义为抽象类,抽象类也要使用“abstract”关键字来修饰;在抽象类里面可以有不是抽象的方法和成员属性,但只要有一个方法是抽象的方法,这个类就必须声明为抽象类,使用”abstract”来修饰。例如:

 代码如下 复制代码

abstract class Demo
{
    var $test;

    abstract function fun1();
    abstract function fun2();
    function fun3()
    {
        ... ...
    }
}

上例中定义了一个抽象类“Demo”使用了”abstract”来修饰, 在这个类里面定义了一个成员属性“$test”,和两个抽象方法“fun1”和“fun2”还有一个非抽象的方法fun3();那么抽象类我们怎么使用呢?最重要的一点就是抽象类不能产生实例对象,所以也不能直接使用,前面我们多次提到过类不能直接使用,我们使用的是通过类实例化出来的对象,那么抽象类不能产生实例对象我们声明抽象类有什么用呢?我们是将抽象方法是做为子类重载的模板使用的,定义抽象类就相当于定义了一种规范,这种规范要求子类去遵守,子类继函抽象类之后,把抽象类里面的抽象方法按照子类的需要实现。子类必须把父类中的抽象方法全部都实现,否则子类中还存在抽象方法,那么子类还是抽象类,还是不能实例化对;为什么我们非要从抽象类中继承呢?因为有的时候我们要实现一些功能就必须从抽象类中继承,否则这些功能你就实现不了,如果继承了抽象类,就要实现类其中的抽象方法;

 代码如下 复制代码

abstract class Demo
    {
    var $test;

    abstract function fun1();
    abstract function fun2();
    function fun3()
    {
        ... ...
    }
}

//抽象类为能产生实例对象,所以这样做是错的,实例化对象交给子类

 代码如下 复制代码

$demo=new Demo();

class Test extends Demo
{
    function fun1()
    {
        ... ...
    }

    function fun2()
    {
        ... ...
    }
}

//子类可以实例化对象,因为实现了父类中所有抽象方法

 代码如下 复制代码
$test=new Test();

__call处理调用错误


在程序开发中,如果在使用对象调用对象内部方法时候,调用的这个方法不存在那么程序就会出错,然后程序退出不能继续执行。那么可不可以在程序调用对象内部不存在的方法时,提示我们调用的方法及使用的参数不存在,但程序还可以继续执行,这个时候我们就要使用在调用不存在的方法时自动调用的方法”__call()”。

 代码如下 复制代码

//这是一个测试的类,里面没有属性和方法
class Test
{
}

//产生一个Test类的对象
$test=new Test();

//调用对象里不存在的方法
$test->demo("one", "two", "three");

//程序不会执行到这里
echo "this is a test";

上例出现如下错误,程序通出不能继续执行;
Fatal error: Call to undefined method Test::demo()

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

 代码如下 复制代码

//这是一个测试的类,里面没有属性和方法
class Test
{
    //调用不存的方法时自动调用的方法,第一个参数为方法名,第二个参数是数组参数
    function __call($function_name, $args)
    {
        print "你所调用的函数:$function_name(参数:";
        print_r($args);
        print ")不存在!n";
    }
}

//产生一个Test类的对象
$test=new Test();

//调用对象里不存在的方法
$test->demo("one", "two", "three");

//程序不会退出可以执行到这里
echo "this is a test";

上例输出结果为:
你所调用的函数: demo(参数:Array ( [0] => one [1] => two [2] => three ) )不存在! this is a test.


克隆对象


有的时候我们需要在一个项目里面,使用两个或多个一样的对象,如果你使用“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."";
    }
}

$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."";
    }

    //对象克隆时自动调用的方法, 如果想在克隆后改变原对象的内容,需要在__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

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/629205.htmlTechArticle本文章介绍了关于php中面向对象的抽象方法和抽象类 __call 克隆对象用法,有需要发解的朋友可以学学。 抽象方法和抽象类 在OOP语言中,一...
Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn