抽象类和接口的区别是php面试经常被问到的问题,也是一个经典问题。我们尽量引用官方权威的说明或者经过实验来证明本文所说的内容准确性。
一、抽象类
官方文档描述:
定义为抽象的类不能被实例化。任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的。(抽象类可以没有抽象方法,但是抽象类依然不能被实例化)被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。 如
abstract class AbstractClass { // 强制要求子类定义这些方法,且不能定义具体功能 注意没有大括号{} abstract protected function getValue (); abstract protected function prefixValue ( $prefix ); // 普通方法(非抽象方法) public function printOut () { print $this -> getValue () . "\n" ; } }
抽象类的特点:
继承一个抽象类的时候,非抽象子类必须定义父类中的所有抽象方法;另外,这些方法的访问控制必须和父类中一样(或者更为宽松)。例如某个抽象方法被声明为受保护的,那么子类中实现的方法就应该声明为受保护的或者公有的,而不能定义为私有的。
此外方法的调用方式必须匹配,即类型和所需参数数量必须一致。例如,子类定义了一个可选参数(类似function eat($a,$b=1)中的$b就是可选参数),而父类抽象方法的声明里没有,则两者的声明并无冲突。这也适用于 PHP 5.4 起的构造函数。在 PHP 5.4 之前的构造函数声明可以不一样的。
其他特点:
抽象类可以有成员属性。
有人问:抽象方法是否可以定义为私有,答案是不可以,因为抽象接口的目的就是为了抽象出类模型用来继承,定义为私有,外部访问不到,偏移了设计目的。如下会报错
抽象类可以实现接口,且可以不实现其中的方法(后面接口的实现中有代码)
<?php abstract class Sutdent extends Human { abstract private function study(); }
抽象类可以继承抽象类,且不能重写抽象父类的抽象方法。这样的用法,可以理解为对抽象类的扩展。如
<?php abstract class Human { abstract function eat(); } abstract class Sutdent extends Human { abstract function eat();// 这不是重写 } abstract class Sutdent2 extends Human { abstract function eat($foot); //重写 eat() 方法 }
二、接口
接口的定义:
使用接口(interface),可以指定某个类必须实现哪些方法,但不需要定义这些方法的具体内容。
接口是通过 interface 关键字来定义的,就像定义一个标准的类一样,但其中定义所有的方法都是空的。
接口中定义的所有方法都必须是公有,这是接口的特性,protected和private会报错(Fatal error: Access type for interface method)。
常量:接口中也可以定义常量。接口常量和类常量的使用完全相同,但是不能被子类或子接口所覆盖。(不建议这样用,实在想不到有什么意义,也容易产生和抽象类的混淆)
interface Play { const LEVEL=10; public function PlayLOL(); public function PlayFootball(); }
接口的实现:
要实现一个接口,使用 implements 操作符。非抽象类中必须实现接口中定义的所有方法,否则会报一个致命错误。类可以实现多个接口,用逗号来分隔多个接口的名称。
接口的特点:
可以同时继承抽象类和实现接口,extends要写在前面.
抽象类实现接口,不需要重新其中的方法。
实现多个接口时,接口中的方法不能有重名。
接口也可以继承,通过使用 extends 操作符。
类要实现接口,必须使用和接口中所定义的方法完全一致的方式。否则会导致致命错误。
interface Play { const LEVEL=10; public function PlayLOL(); public function PlayFootball(); } interface Read { public function ReadNovel(); } abstract class Human{ abstract function eat(); } //抽象类可以实现接口后不实现其方法,可以继承一个抽象类的同时实现多个接口注意必须要把extends语句写在implements前面,否则会报错 abstract class Sutdent extends Human implements Play,Read{ abstract function study(); } //implements在extends前面 abstract class Sutdent2 implements Play,Read extends Human { abstract function study(); }
接口中的所有方法都不能有方法体:
<?php //所有方法都没有方法体 interface Play { public function PlayFootball(); public function way(); } interface PlayNew { public function PlayFootballNew(); // way 方法有方法体 public function way() { echo "way"; } }
运行结果:
接口继承
接口可以继承另一个或多个接口,使用extends关键字,多个用 ',' 隔开,但是不能实现另一个接口,当然更不能继承抽象类(继承抽象类报错)
<?php interface Play { public function PlayFootball(); } interface PlayNew { public function PlayFootballNew(); } interface PlayGame extends play,PlayNew { public function PlayLOL(); } abstract class Human { abstract function eat(); } interface PlayGame2 extends Human { public function eat(); }
php抽象类和接口的区别总结:
1、对接口的继承使用implements,抽象类使用extends;
2、接口中不可以声明变量,但可以声明类常量.抽象类中可以声明各种变量;
3、接口没有构造函数,抽象类可以有;
4、接口中的方法默认为public,抽象类中的方法可以用public,protected,private修饰;
5、一个类可以继承多个接口,但只能继承一个抽象类。