php中模拟多继承如何实现

php中模拟多继承如何实现

一、总结

一句话总结:其实你继承别人也是想调用别人类里面的方法和属性,所以可以这样做:这本类中创建目标类的对象,然后通过这个对象来调用方法和属性,这样比继承来的方便。

二、魔术方法 __call 实现模拟多继承

1、相关知识

PHP没有多继承的特性。即使是一门支持多继承的编程语言,我们也很少会使用这个特性。在大多数人看来,多继承不是一种好的设计方法。想要给某个类添加额外的特性,不一定要使用继承。这里我提供一种模拟多继承的方法以供参考。

PHP有一个魔术方法,叫做__call。当你调用一个不存在的方法时,这个方法会被自动调用。这时,我们就有机会将调用重定向到一个存在的方法。继承多个父类的子类,寻找方法的过程一般是这样的:

本身的方法 -> 父类1的方法 -> 父类2的方法...

模拟过程大致是这样:将各个父类实例化,然后作为子类的属性。这些父类提供一些公有的方法。当子类拥有某方法时,__call()函数不会被调用。这相当于“覆盖”了父类的方法。当调用了不存在的方法时,通过__call()方法依次从父类中寻找可以调用的方法。虽然这不是完备的多继承,但可以帮助我们解决问题。

 1 <?php
 2 class Parent1 {
 3     function method1() {}
 4     function method2() {}
 5 }
 6 class Parent2 {
 7     function method3() {}
 8     function method4() {}
 9 }
10 class Child {
11     protected $_parents = array();
12     public function Child(array $parents=array()) {
13         $_parents = $parents;
14     }
15      
16     public function __call($method, $args) {
17         // 从“父类"中查找方法
18         foreach ($this->_parents as $p) {
19             if (is_callable(array($p, $method))) {
20                 return call_user_func_array(array($p, $method), $args);
21             }
22         }
23         // 恢复默认的行为,会引发一个方法不存在的致命错误
24         return call_user_func_array(array($this, $method), $args);
25     }
26 }
27 $obj = new Child(array(new Parent1(), new Parent2()));
28 $obj->method1();
29 $obj->method3();

这里没有涉及属性的继承,但实现起来并不困难。可以通过__set()和__get()魔术方法来模拟属性的继承。请你动手实践。

其它方法:通过接口来实现多继承

php中的类只能继承一个父类,如果要继承多个类应采用接口

interface 模拟多继承

三、PHP接口实现多继承详解

在PHP的接口中,接口可以继承接口。虽然PHP类只能继承一个父类(单继承),但是接口和类不同,接口可以实现多继承,可以继承一个或者多个接口。当然接口的继承也是使用extends关键字,要多个继承的话只要用逗号把继承的接口隔开即可。


   需要注意的是当你接口继承其它接口时候,直接继承父接口的静态常量属性和抽象方法,所以类实现接口时必须实现所有相关的抽象方法

   现在你对PHP接口的继承有所了解了吧,下面的例子可供参考,代码如下:

  1. <?php 
  2. interface father{ 
  3. function shuchu(); 
  4. }  
  5. interface fam extends father{ 
  6. function cook($name); 
  7. }  
  8. class test implements fam{ 
  9. function shuchu(){ 
  10. echo "接口继承,要实现两个抽象方法"; 
  11. echo "<br>"; 
  12. }  
  13. function cook($name){ 
  14. echo "平时经常做饭的人是:".$name;  
  15. }  
  16. }  
  17. $t=new test(); 
  18. $t->shuchu(); 
  19. $t->cook("妈妈");  
  20. ?> 

代码运行结果如下:

接口继承,要实现两个抽象方法
平时经常做饭的人是:妈妈
 


上面的示例是接口继承了一个接口,所以在test类实现fam接口的时候要实例两个抽象方法,就是把接口的子类和父类的抽象方法都实例。


下面来看一个接口多继承的示例,代码如下:

  1. <?php 
  2. interface father{ 
  3. function shuchu(); 
  4. interface mother{ 
  5. function dayin($my); 
  6. interface fam extends father,mother{ 
  7. function cook($name); 
  8. class test implements fam{ 
  9. function dayin($my){ 
  10. echo "我的名字是:".$my;  
  11. echo "<br>"; 
  12. function shuchu(){ 
  13. echo "接口继承,要实现两个抽象方法"; 
  14. echo "<br>"; 
  15. function cook($name){ 
  16. echo "平时经常做饭的人是:".$name;  
  17. $t=new test(); 
  18. $t->shuchu(); 
  19. $t->dayin("小强");  
  20. $t->cook("妈妈");  
  21. ?> 


示例运行结果:

接口继承,要实现两个抽象方法
我的名字是:小强
平时经常做饭的人是:妈妈
 


这段代码由于接口继承了两个接口,所有实例时要把这三个抽象类的所有抽象方法都实例,总共有三个。看完这两个例子,你应该对接口的继承熟悉了吧,其实就一个单继承和多继承,只要实现了所有相关的抽象方法就可以了。

 
原文地址:https://www.cnblogs.com/Renyi-Fan/p/9266322.html