php面向对象抽象方法和抽象类 __call 克隆对象详细教程
本文章介绍了关于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