>  기사  >  백엔드 개발  >  PHP의 추상 클래스와 인터페이스의 정의와 사용법에 대한 자세한 설명

PHP의 추상 클래스와 인터페이스의 정의와 사용법에 대한 자세한 설명

伊谢尔伦
伊谢尔伦원래의
2017-07-03 09:47:582083검색

1. 추상수업추상수업

1. 추상 클래스는 클래스 앞에 abstract 키워드가 있고 추상 메서드(클래스 메서드 function 키워드 앞에 abstract 키워드)가 있는 클래스를 말합니다.

2. 추상 클래스는 직접 인스턴스화할 수 없습니다. 추상 클래스는 하위 클래스에 필요한 메서드만 정의(또는 부분적으로 구현)합니다. 서브클래스는 추상 클래스를 상속하고 추상 클래스의 모든 추상 메서드를 구현하여 추상 클래스를 구체적으로 만들 수 있습니다.

3. 하위 클래스를 인스턴스화해야 하는 경우 추상 클래스에 모든 추상 메서드를 구현해야 합니다. 서브클래스가 추상 클래스의 모든 추상 메소드를 구현하지 않는 경우 서브클래스도 추상 클래스입니다. abstract 키워드는 클래스 앞에 추가되어야 하며 인스턴스화될 수 없습니다.

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";  
    }  
}

4. 아래와 같이 A에서 상속되는 하위 클래스 B를 생성하지만 추상 메소드 abstract_func():

Class B extends A{};

를 구현하지 않으면 프로그램에 다음 오류가 발생합니다:

Fatal error: Class B contains 1 abstract method and must therefore be declared abstract or implement the remaining methods (A::abstract_func)

5. B가 추상 메소드 abstract_func()를 구현하는 경우 B에 있는 abstract_func() 메소드의 액세스 제어는 A에 있는 abstract_func()의 액세스 제어보다 엄격할 수 없습니다. 즉,

(1) A()의 abstract_func 선언이 공개이므로 b의 abstract_func() 문은 공개만 가능하고 프로젝트나 비공개는 불가능합니다

(2) A의 abstract_func() 문이 profestru인 경우 공개입니다. 또는 Protected이지만 비공개일 수 없습니다

(3) A의 Abstract_func()가 Private로 선언되면 비공개로 정의할 수 없습니다! (치명적인 오류: 추상 함수 A::abstract_func()는 declared 비공개일 수 없습니다.)

2 인터페이스 인터페이스

1. 추상 클래스는 구체적인 구현을 위한 표준을 제공하는 반면 인터페이스는 순수한 템플릿입니다. 인터페이스는 구현 내용이 아닌 기능만 정의합니다. 인터페이스는 키워드 인터페이스를 사용하여 선언됩니다.

2. 인터페이스는 완전히 추상적입니다. 메서드만 선언할 수 있고, 공용 메서드만 선언할 수 있으며, 메서드 본문을 정의할 수 없고, 인스턴스 변수를 선언할 수도 없습니다. 그러나 인터페이스는

상수변수를 선언할 수 있습니다. 그러나 인터페이스에 상수 변수를 배치하는 것은 인터페이스로서의 존재 목적에 위배되며 인터페이스와 클래스의 서로 다른 값을 혼동하기도 합니다. 꼭 필요한 경우 해당 추상 클래스나 클래스에 넣을 수 있습니다.

interface iA  
{  
    const AVAR=3;  
    public function iAfunc1();  
    public function iAfunc2();  
}  
echo iA:: AVAR;

3. 인터페이스를 구현하는 모든 클래스는 인터페이스

class E implements iA  
{  
    public function iAfunc1(){echo "in iAfunc1";}  
    public function iAfunc2(){echo "in iAfunc2";}  
}

에 정의된 모든 메서드를 구현해야 합니다. 그렇지 않으면 클래스를 abstract로 선언해야 합니다.

abstract class E implements iA{}

4.一个类可以在声明中使用implements关键字来实现某个接口。这么做之后,实现接口的具体过程和继承一个仅包含抽象方法的抽象类是一样的。一个类可以同时继承一个父类和实现任意多个接口。extends子句应该在implements子句之前。PHP只支持继承自一个父类,因此extends关键字后只能跟一个类名。

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";}  
}

5.接口不可以实现另一个接口,但可以继承多个

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";}  
}

三、抽象类和接口的异同

1.相同点:

(1)     两者都是抽象类,都不能实例化。

(2)     interface实现类及abstract class的子类都必须要实现已经声明的抽象方法。

2. 不同点:

(1)     interface需要实现,要用implements,而abstract class需要继承,要用extends。

(2)     一个类可以实现多个interface,但一个类只能继承一个abstract class。

(3)     interface强调特定功能的实现,而abstract class强调所属关系。

(4)     尽管interface实现类及abstract class的子类都必须要实现相应的抽象方法,但实现的形式不同。interface中的每一个方法都是抽象方法,都只是声明的 (declaration, 没有方法体),实现类必须要实现。而abstract class的子类可以有选择地实现。这个选择有两点含义:a) abstract class中并非所有的方法都是抽象的,只有那些冠有abstract的方法才是抽象的,子类必须实现。那些没有abstract的方法,在 abstract class中必须定义方法体;b) abstract class的子类在继承它时,对非抽象方法既可以直接继承,也可以覆盖;而对抽象方法,可以选择实现,也可以留给其子类来实现,但此类必须也声明为抽象类。既是抽象类,当然也不能实例化。

(5)     abstract class是interface与class的中介。abstract class在interface及class中起到了承上启下的作用。一方面,abstract class是抽象的,可以声明抽象方法,以规范子类必须实现的功能;另一方面,它又可以定义缺省的方法体,供子类直接使用或覆盖。另外,它还可以定义自己的实例变量,以供子类通过继承来使用。

(6)     接口中的抽象方法前不用也不能加abstract关键字,默认隐式就是抽象方法,也不能加final关键字来防止抽象方法的继承。而抽象类中抽象方法前则必须加上abstract表示显示声明为抽象方法。

(7)     接口中的抽象方法默认是public的,也只能是public的,不能用private,protected修饰符修饰。而抽象类中的抽象方法则可以用public,protected来修饰,但不能用private。

3. interface的应用场合

(1)     类与类之间需要特定的接口进行协调,而不在乎其如何实现。

(2)     作为能够实现特定功能的标识存在,也可以是什么接口方法都没有的纯粹标识。

(3)     需要将一组类视为单一的类,而调用者只通过接口来与这组类发生联系。

(4)     需要实现特定的多项功能,而这些功能之间可能完全没有任何联系。

4. abstract class的应用场合

一句话,在既需要统一的接口,又需要实例变量或缺省的方法的情况下,就可以使用它。最常见的有:

(1)     定义了一组接口,但又不想强迫每个实现类都必须实现所有的接口。可以用abstract class定义一组方法体,甚至可以是空方法体,然后由子类选择自己所感兴趣的方法来覆盖。

(2)     某些场合下,只靠纯粹的接口不能满足类与类之间的协调,还必需类中表示状态的变量来区别不同的关系。abstract的中介作用可以很好地满足这一点。

(3) 상호 조정된 메서드 집합을 표준화합니다. 그 중 일부는 공통적이고 상태 독립적이며 하위 클래스가 별도로 구현할 필요 없이 공유할 수 있지만 다른 메서드에서는 각 하위 클래스가 해당 메서드를 구현해야 합니다. 특정 기능을 구현하기 위해 특정 상태를 소유함 .

위 내용은 PHP의 추상 클래스와 인터페이스의 정의와 사용법에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.