博客列表 >抽象类与接口论述

抽象类与接口论述

后网络时代
后网络时代原创
2021年03月13日 02:08:13595浏览

自己定义一个接口和抽象类,并实现它,对比接口与抽象类的区别与联系,并实例演示
答:

  1. <?php
  2. // 使用接口(interface),可以指定某个类必须实现哪些方法,但不需要定义这些方法的具体内容。
  3. // 接口是通过 interface 关键字来定义的,就像定义一个标准的类一样,但其中定义所有的方法都是空的。
  4. // 接口中定义的所有方法都必须是公有,这是接口的特性。
  5. // 需要注意的是,在接口中定义一个构造方法是被允许的。在有些场景下这可能会很有用,例如用于工厂模式时。
  6. // 接口中也可以定义常量。接口常量和类常量的使用完全相同,但是不能被子类或子接口所覆盖。
  7. interface face1{
  8. public const name=1111;
  9. }
  10. interface face2 extends face1 {
  11. const name1="接口的多继承";
  12. }
  13. echo "<hr>";
  14. interface middleinput{
  15. //接口中,能定义两类成员:类常量, 公共的方法
  16. const app_name="读书商城";
  17. public static function m1():string;
  18. //构造方法
  19. public function __construct(...$arg);
  20. }
  21. //要实现一个接口,使用 implements 操作符。类中必须实现接口中定义的所有方法,否则会报一个致命错误。类可以实现多个接口,用逗号来分隔多个接口的名称。
  22. // 在 PHP 5.3.9 之前,实现多个接口时,接口中的方法不能有重名,因为这可能会有歧义。在最近的 PHP 版本中,只要这些重名的方法签名相同,这种行为就是允许的。
  23. // 接口也可以继承,通过使用 extends 操作符。
  24. class Demo implements middleinput{
  25. public static function m1():string{
  26. return "---------------------------";
  27. }
  28. public function __construct(...$arg){
  29. echo "带参数构造方法";
  30. }
  31. }
  32. echo "************************";
  33. echo Demo::m1();
  34. new Demo;
  35. echo "<hr>";
  36. // 类对接口的实现也是多实现的
  37. class ce implements face1,face2{
  38. function __construct()
  39. {
  40. }
  41. }
  42. echo ce::name1,"<br>";
  43. //接口继承:接口却是多继承
  44. interface test1{
  45. public function __construct($name);
  46. }
  47. interface test2{
  48. public function __construct($name);
  49. }
  50. class Tclass implements test1,test2{
  51. public function __construct($name){
  52. echo "OK".$name;
  53. }
  54. }
  55. new Tclass('接口中同名的方法,在实现的时候根据需要去实现!');
  56. echo "<hr>";
  57. // PHP 5 支持抽象类和抽象方法。定义为抽象的类不能被实例化。
  58. // 任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的。
  59. // 被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。
  60. // 继承一个抽象类的时候,子类必须定义父类中的所有抽象方法;
  61. // 另外,这些方法的访问控制必须和父类中一样(或者更为宽松)。
  62. // 例如某个抽象方法被声明为受保护的,那么子类中实现的方法就应该声明为受保护的或者公有的,
  63. // 而不能定义为私有的。此外方法的调用方式必须匹配,即类型和所需参数数量必须一致。
  64. // 例如,子类定义了一个可选参数,而父类抽象方法的声明里没有,则两者的声明并无冲突。
  65. // 这也适用于 PHP 5.4 起的构造函数。在 PHP 5.4 之前的构造函数声明可以不一样的。
  66. abstract class AbstractClass{
  67. var $name; //可以定义变量
  68. // 我们的抽象方法仅需要定义需要的参数
  69. abstract protected function prefixName($name);
  70. }
  71. class abstractclassed extends AbstractClass{
  72. //重写父类中的方法,访问权限必须和父类中的一致或者比父类的访问权限更大
  73. //参数必须包含父类中的参数次序一致,子类可以定义父类签名中不存在的可选参数
  74. public function prefixName($name,$spotor="."){
  75. echo $name;
  76. }
  77. }
  78. $obj=new abstractclassed;
  79. $obj->prefixName('可选参数');
  80. // 一个类可以在声明中使用 implements 关键字来实现某个接口。
  81. // 这么做之后,实现接口的具体过程和继承一个仅包含抽象方法的抽象类是一样的。
  82. // 一个类可以同时继承一个父类和实现任意多个接口。
  83. // extends 子句应该在 implements 子句之前。
  84. // PHP 只支持继承自一个父类,因此 extends 关键字后只能跟一个类名。
  85. interface iB
  86. {
  87. public function iBfunc1();
  88. public function iBfunc2();
  89. }
  90. class D extends A implements iB
  91. {
  92. public function abstract_func1()
  93. {
  94. echo "implement the abstract_func1 in class A/n";
  95. }
  96. public function abstract_func2()
  97. {
  98. echo "implement the abstract_func2 in class A/n";
  99. }
  100. public function iAfunc1(){echo "in iAfunc1";}
  101. public function iAfunc2(){echo "in iAfunc2";}
  102. public function iBfunc1(){echo "in iBfunc1";}
  103. public function iBfunc2(){echo "in iBfunc2";}
  104. }
  105. echo "<hr>";
  106. //抽象类是指被abstract修饰的类
  107. abstract class A
  108. {
  109. }
  110. // PHP 5 支持抽象类和抽象方法。定义为抽象的类不能被实例化。
  111. // 任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的。
  112. // 被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。
  113. // 继承一个抽象类的时候,子类必须定义父类中的所有抽象方法;
  114. // 另外,这些方法的访问控制必须和父类中一样(或者更为宽松)。
  115. // 例如某个抽象方法被声明为受保护的,那么子类中实现的方法就应该声明为受保护的或者公有的,
  116. // 而不能定义为私有的。此外方法的调用方式必须匹配,即类型和所需参数数量必须一致。
  117. // 例如,子类定义了一个可选参数,而父类抽象方法的声明里没有,则两者的声明并无冲突。
  118. // 这也适用于 PHP 5.4 起的构造函数。在 PHP 5.4 之前的构造函数声明可以不一样的。
  119. abstract class AbstractClass{
  120. // 我们的抽象方法仅需要定义需要的参数
  121. abstract protected function prefixName($name);
  122. }
  123. class abstractclassed extends AbstractClass{
  124. //重写父类中的方法,访问权限必须和父类中的一致或者比父类的访问权限更大
  125. //参数必须包含父类中的参数次序一致,子类可以定义父类签名中不存在的可选参数
  126. public function prefixName($name,$spotor="."){
  127. echo $name;
  128. }
  129. }
  130. $obj=new abstractclassed;
  131. $obj->prefixName('可选参数');
  132. echo "<hr>";
  133. // 总结:
  134. // 抽象类和接口的异同
  135. // 1. 相同点:
  136. // (1) 两者都是抽象类,都不能实例化。
  137. // (2)interface的 实现类及 abstract class 的子类都必须要实现已经声明的抽象方法。
  138. // 2. 不同点:
  139. // (1) interface 需要实现,要用 implements ,而 abstract class 需要继承,要用 extends 。
  140. // (2)一个类可以实现多个 interface ,但一个类只能继承一个 abstract class 。
  141. // (3)interface强调特定功能的实现,而 abstract class 强调所属关系和特定功能的实现。
  142. // (4)尽管 interface 实现类及 abstract class 的子类都必须要实现相应的抽象方法,但实现的形式不同。 interface 中的每一个方法都是抽象方法,都只是声明的 (declaration, 没有方法体 ) ,实现类必须要实现。而 abstract class 的子类可以有选择地实现。这个选择有两点含义: a) abstract class 中并非所有的方法都是抽象的,只有那些冠有 abstract 的方法才是抽象的,子类必须实现。那些没有 abstract 的方法,在 abstract class 中必须定义方法体; b) abstract class 的子类在继承它时,对非抽象方法既可以直接继承,也可以覆盖;而对抽象方法,可以选择实现,也可以留给其子类来实现,但此类必须也声明为抽象类。既是抽象类,当然也不能实例化。
  143. // (5)abstract class 是 interface 与 class 的中介。 abstract class 在 interface 及 class 中起到了承上启下的作用。一方面, abstract class 是抽象的,可以声明抽象方法,以规范子类必须实现的功能;另一方面,它又可以定义缺省的方法体,供子类直接使用或覆盖。另外,它还可以定义自己的实例变量,以供子类通过继承来使用。
  144. // (6)接口中的抽象方法前不用也不能加 abstract 关键字,默认隐式就是抽象方法,也不能加 final 关键字来防止抽象方法的继承。而抽象类中抽象方法前则必须加上 abstract 表示显示声明为抽象方法。
  145. // (7)接口中的抽象方法默认是 public 的,也只能是 public 的,不能用 private , protected 修饰符修饰。而抽象类中的抽象方法则可以用 public , protected 来修饰,但不能用 private 。
  146. // 3. interface 的应用场合
  147. // (1)类与类之间需要特定的接口进行协调,而不在乎其如何实现。
  148. // (2)作为能够实现特定功能的标识存在,也可以是什么接口方法都没有的纯粹标识。
  149. // (3)需要将一组类视为单一的类,而调用者只通过接口来与这组类发生联系。
  150. // (4)需要实现特定的多项功能,而这些功能之间可能完全没有任何联系。
  151. // 4. abstract class 的应用场合
  152. // 一句话,在既需要统一的接口,又需要实例变量或缺省的方法的情况下,就可以使用它。最常见的有:
  153. // (1)定义了一组接口,但又不想强迫每个实现类都必须实现所有的接口。可以用 abstract class 定义一组方法体,甚至可以是空方法体,然后由子类选择自己所感兴趣的方法来覆盖。
  154. // (2)某些场合下,只靠纯粹的接口不能满足类与类之间的协调,还必需类中表示状态的变量来区别不同的关系。 abstract 的中介作用可以很好地满足这一点。
  155. // (3)规范了一组相互协调的方法,其中一些方法是共同的,与状态无关的,可以共享的,无需子类分别实现;而另一些方法却需要各个子类根据自己特定的状态来实现特 定的功能 。

效果:

声明:本文内容转载自脚本之家,由网友自发贡献,版权归原作者所有,如您发现涉嫌抄袭侵权,请联系admin@php.cn 核实处理。
全部评论
文明上网理性发言,请遵守新闻评论服务协议