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 비공개일 수 없습니다.)
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 중국어 웹사이트의 기타 관련 기사를 참조하세요!