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

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

抽象方法和抽象类

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

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

实例代码如下:

  1. abstract function fun1();
  2. abstract function fun2();

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

实例代码如下:

  1. abstract class Demo
  2. {
  3. var $test;
  4. abstract function fun1();
  5. abstract function fun2();
  6. function fun3()
  7. {
  8. ... ...
  9. }
  10. }

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

实例代码如下:

  1. abstract class Demo
  2. {
  3. var $test;
  4. abstract function fun1();
  5. abstract function fun2();
  6. function fun3()
  7. {
  8. ... ...
  9. }
  10. }
  11. //抽象类为能产生实例对象,所以这样做是错的,实例化对象交给子类

实例代码如下:

  1. $demo=new Demo();
  2. class Test extends Demo
  3. {
  4. function fun1()
  5. {
  6. ... ...
  7. }
  8. function fun2()
  9. {
  10. ... ...
  11. }
  12. }
  13. //子类可以实例化对象,因为实现了父类中所有抽象方法

实例代码如下:

  1. $test=new Test();

__call处理调用错误

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

实例代码如下:

  1. //这是一个测试的类,里面没有属性和方法
  2. class Test
  3. {
  4. }
  5. //产生一个Test类的对象
  6. $test=new Test();
  7. //调用对象里不存在的方法
  8. $test->demo("one", "two", "three");
  9. //程序不会执行到这里
  10. echo "this is a test";
  11. 上例出现如下错误,程序通出不能继续执行;
  12. Fatal error: Call to undefined method Test::demo()

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

实例代码如下:

  1. //这是一个测试的类,里面没有属性和方法
  2. class Test
  3. {
  4. //调用不存的方法时自动调用的方法,第一个参数为方法名,第二个参数是数组参数
  5. function __call($function_name, $args)
  6. {
  7. print "你所调用的函数:$function_name(参数:";
  8. print_r($args);
  9. print ")不存在!n";
  10. }
  11. }
  12. //产生一个Test类的对象
  13. $test=new Test();
  14. //调用对象里不存在的方法
  15. $test->demo("one", "two", "three");
  16. //程序不会退出可以执行到这里
  17. echo "this is a test";

上例输出结果为:

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

克隆对象

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

在PHP5中我们使用”clone”这个关键字克隆对象;

实例代码如下:

  1. class Person
  2. {
  3. //下面是人的成员属性
  4. var $name; //人的名子
  5. var $sex; //人的性别
  6. var $age; //人的年龄
  7. //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值
  8. function __construct($name="", $sex="", $age="")
  9. {
  10. $this->name=$name;
  11. $this->sex=$sex;
  12. $this->age=$age;
  13. }
  14. //这个人可以说话的方法, 说出自己的属性
  15. function say()
  16. {
  17. echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."";
  18. }
  19. }
  20. $p1=new Person("张三", "男", 20);
  21. //使用“clone”克隆新对象p2,和p1对象具有相同的属性和方法.
  22. $p2=clone $p1;
  23. $p2->say();

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

实例代码如下:

  1. class Person
  2. {
  3. //下面是人的成员属性
  4. var $name; //人的名子
  5. var $sex; //人的性别
  6. var $age; //人的年龄
  7. //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值
  8. function __construct($name="", $sex="", $age="")
  9. {
  10. $this->name=$name;
  11. $this->sex=$sex;
  12. $this->age=$age;
  13. }
  14. //这个人可以说话的方法, 说出自己的属性
  15. function say()
  16. {
  17. echo "我的名子叫:".$this->name." 性别:".$this->sex." 我的年龄是:".$this->age."";
  18. }
  19. //对象克隆时自动调用的方法, 如果想在克隆后改变原对象的内容,需要在__clone()中重写原本的属性和方法
  20. function __clone()
  21. {
  22. //$this指的复本p2, 而$that是指向原本p1,这样就在本方法里,改变了复本的属性.
  23. $this->name="我是假的$that->name";
  24. $this->age=30;
  25. }
  26. }
  27. $p1=new Person("张三", "男", 20);
  28. $p2=clone $p1;
  29. $p1->say();
  30. $p2->say();

上例输出:

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

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