PHP中的 抽象类(abstract class)和 接口(interface)

抽象类abstract class

  • 抽象类是指在class前加了abstract关键字且存在抽象方法(在类方法function关键字前加了abstract关键字)的类。
  • 抽象类不能被直接实例化。抽象类中只定义(或部分实现)子类需要的方法。子类可以通过继承抽象类并通过实现抽象类中的所有抽象方法,使抽象类具体化。
  • 如果子类需要实例化,前提是它实现了抽象类中的所有抽象方法。如果子类没有全部实现抽象类中的所有抽象方法,那么该子类也是一个抽象类,必须在class前面加上abstract关键字,并且不能被实例化。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    abstract class A  
    {
    /** 抽象类中可以定义变量 */
    protected $value1 = 0;
    private $value2 = 1;
    public $value3 = 2;
    /** 也可以定义非抽象方法 */
    public function my_print()
    {
    echo "hello,world/n";
    }
    /**
    * 大多数情况下,抽象类至少含有一个抽象方法。抽象方法用abstract关键字声明,其中不能有具体内容。
    * 可以像声明普通类方法那样声明抽象方法,但是要以分号而不是方法体结束。也就是说抽象方法在抽象类中不能被实现,也就是没有函数体“{some codes}”。
    */
    abstract protected function abstract_func1();
    abstract protected function abstract_func2();
    }
    abstract class B extends A
    {
    public function abstract_func1()
    {
    echo "implement the abstract_func1 in class A/n";
    }
    /** 这么写在zend studio 8中会报错*/
    //abstract protected function abstract_func2();
    }
    class C extends B
    {
    public function abstract_func2()
    {
    echo "implement the abstract_func2 in class A/n";
    }
    }
  • 如果像下面这样创建了一个继承自A的子类B,但是不实现抽象方法abstract_func():

    1
    2
    3
    4
    5
    Class B extends A{}; 
    ```
    那么程序将出现以下错误:
    ```php
    Fatal error: Class B contains 1 abstract method and must therefore be declared abstract or implement the remaining methods (A::abstract_func)
  • 如果B实现了抽象方法abstract_func(),那么B中abstract_func()方法的访问控制不能比A中abstract_func()的访问控制更严格,也就是说:
    (1)如果A中abstract_func()声明为public,那么B中abstract_func()的声明只能是public,不能是protected或private
    (2)如果A中abstract_func()声明为protected,那么B中abstract_func()的声明可以是public或protected,但不能是private
    (3)如果A中abstract_func()声明为private,嘿嘿,不能定义为private哦!(Fatal error: Abstract function A::abstract_func() cannot be declared private)

    接口interface

  • 抽象类提供了具体实现的标准,而接口则是纯粹的模版。接口只定义功能,而不包含实现的内容。接口用关键字interface来声明。
  • interface是完全抽象的,只能声明方法,而且只能声明public的方法,不能声明private及protected的方法,不能定义方法体,也不能声明实例变量。然而,interface却可以声明常量变量。但将常量变量放在interface中违背了其作为接口的作用而存在的宗旨,也混淆了interface与类的不同价值。如果的确需要,可以将其放在相应的abstract class或Class中。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
     interface iA  
    {
    const AVAR=3;
    public function iAfunc1();
    public function iAfunc2();
    }
    echo iA:: AVAR;
    ```
    * 任何实现接口的类都要实现接口中所定义的所有方法
    ```php
    class E implements iA
    {
    public function iAfunc1(){echo "in iAfunc1";}
    public function iAfunc2(){echo "in iAfunc2";}
    }
    ```
    否则该类必须声明为abstract
    ```php
    abstract class E implements iA{}
  • 一个类可以在声明中使用implements关键字来实现某个接口。这么做之后,实现接口的具体过程和继承一个仅包含抽象方法的抽象类是一样的。一个类可以同时继承一个父类和实现任意多个接口。extends子句应该在implements子句之前。PHP只支持继承自一个父类,因此extends关键字后只能跟一个类名。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    interface iB  
    {
    public function iBfunc1();
    public function iBfunc2();
    }
    class D extends A implements iA,iB
    {
    public function abstract_func1()
    {
    echo "implement the abstract_func1 in class A/n";
    }
    public function abstract_func2()
    {
    echo "implement the abstract_func2 in class A/n";
    }
    public function iAfunc1(){echo "in iAfunc1";}
    public function iAfunc2(){echo "in iAfunc2";}
    public function iBfunc1(){echo "in iBfunc1";}
    public function iBfunc2(){echo "in iBfunc2";}
    }

    class D extends B implements iA,iB
    {
    public function abstract_func1()
    {
    parent::abstract_func1();
    echo "override the abstract_func1 in class A/n";
    }
    public function abstract_func2()
    {
    echo "implement the abstract_func2 in class A/n";
    }
    public function iAfunc1(){echo "in iAfunc1";}
    public function iAfunc2(){echo "in iAfunc2";}
    public function iBfunc1(){echo "in iBfunc1";}
    public function iBfunc2(){echo "in iBfunc2";}
    }
  • 接口不可以实现另一个接口,但可以继承多个

    interface iC extends iA,iB{}  
    class F implements iC  
    {  
       public function iAfunc1(){echo "in iAfunc1";}  
       public function iAfunc2(){echo "in iAfunc2";}  
       public function iBfunc1(){echo "in iBfunc1";}  
       public function iBfunc2(){echo "in iBfunc2";}  
    }
    

    抽象类和接口的异同

    相同点:

  • 两者都是抽象类,都不能实例化。
  • interface实现类及abstract class的子类都必须要实现已经声明的抽象方法。

    不同点:

    • interface需要实现,要用implements,而abstract class需要继承,要用extends。
    • 一个类可以实现多个interface,但一个类只能继承一个abstract class。
    • interface强调特定功能的实现,而abstract class强调所属关系。
    • 尽管interface实现类及abstract class的子类都必须要实现相应的抽象方法,但实现的形式不同。interface中的每一个方法都是抽象方法,都只是声明的 (declaration, 没有方法体),实现类必须要实现。而abstractclass的子类可以有选择地实现。这个选择有两点含义:a) abstract class中并非所有的方法都是抽象的,只有那些冠有abstract的方法才是抽象的,子类必须实现。那些没有abstract的方法,在 abstract class中必须定义方法体;b) abstract class的子类在继承它时,对非抽象方法既可以直接继承,也可以覆盖;而对抽象方法,可以选择实现,也可以留给其子类来实现,但此类必须也声明为抽象类。既是抽象类,当然也不能实例化。
    • abstract class是interface与class的中介。abstract class在interface及class中起到了承上启下的作用。一方面,abstract class是抽象的,可以声明抽象方法,以规范子类必须实现的功能;另一方面,它又可以定义缺省的方法体,供子类直接使用或覆盖。另外,它还可以定义自己的实例变量,以供子类通过继承来使用。
  • 接口中的抽象方法前不用也不能加abstract关键字,默认隐式就是抽象方法,也不能加final关键字来防止抽象方法的继承。而抽象类中抽象方法前则必须加上abstract表示显示声明为抽象方法。
  • 接口中的抽象方法默认是public的,也只能是public的,不能用private,protected修饰符修饰。而抽象类中的抽象方法则可以用public,protected来修饰,但不能用private。

    interface的应用场合

  • 类与类之间需要特定的接口进行协调,而不在乎其如何实现。
  • 作为能够实现特定功能的标识存在,也可以是什么接口方法都没有的纯粹标识。
  • 需要将一组类视为单一的类,而调用者只通过接口来与这组类发生联系。
  • 需要实现特定的多项功能,而这些功能之间可能完全没有任何联系。

    abstract class的应用场合

    一句话,在既需要统一的接口,又需要实例变量或缺省的方法的情况下,就可以使用它。最常见的有:
  • 定义了一组接口,但又不想强迫每个实现类都必须实现所有的接口。可以用abstract class定义一组方法体,甚至可以是空方法体,然后由子类选择自己所感兴趣的方法来覆盖。
  • 某些场合下,只靠纯粹的接口不能满足类与类之间的协调,还必需类中表示状态的变量来区别不同的关系。abstract的中介作用可以很好地满足这一点。
  • 规范了一组相互协调的方法,其中一些方法是共同的,与状态无关的,可以共享的,无需子类分别实现;而另一些方法却需要各个子类根据自己特定的状态来实现特定的功能 。
谢♪翔 wechat
坚持原创技术分享,您的支持将鼓励我继续创作!
-------------本文结束感谢您的阅读-------------
0%