1、 抽象类abstract classphp
1 .抽象类是指在 class 前加了 abstract 关键字且存在抽象方法(在类方法 function 关键字前加了 abstract 关键字)的类。ide
2 .抽象类不能被直接实例化。抽象类中只定义(或部分实现)子类须要的方法。子类能够经过继承抽象类并经过实现抽象类中的全部抽象方法,使抽象类具体化。函数
3 .若是子类须要实例化,前提是它实现了抽象类中的全部抽象方法。若是子类没有所有实现抽象类中的全部抽象方法,那么该子类也是一个抽象类,必须在 class 前面加上 abstract 关键字,而且不能被实例化。spa
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"
;
}
}
|
4 .若是像下面这样建立了一个继承自 A 的子类 B ,可是不实现抽象方法 abstract_func() :code
1
|
Class B
extends
A{};
|
那么程序将出现如下错误:blog
1
|
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() 声明为 public ,那么 B 中 abstract_func() 的声明只能是 public ,不能是 protected 或 private接口
(2) 若是 A 中 abstract_func() 声明为 protected ,那么 B 中 abstract_func() 的声明能够是 public 或 protected ,但不能是 privateci
(3) 若是 A 中 abstract_func() 声明为 private ,嘿嘿,不能定义为 private 哦!( Fatal error : Abstract function A::abstract_func() cannot be declared private )rem
2、 接口interface
1 .抽象类提供了具体实现的标准,而接口则是纯粹的模版。接口只定义功能,而不包含实现的内容。接口用关键字 interface 来声明。
2 . interface 是彻底抽象的,只能声明方法,并且只能声明 public 的方法,不能声明 private 及 protected 的方法,不能定义方法体,也不能声明实例变量 。然而, interface 却能够声明常量变量 。但将常量变量放在 interface 中违背了其做为接口的做用而存在的宗旨,也混淆了 interface 与类的不一样价值。若是的确须要,能够将其放在相应的 abstract class 或 Class 中。
1
2
3
4
5
6
7
|
interface
iA
{
const
AVAR=3;
public
function
iAfunc1();
public
function
iAfunc2();
}
echo
iA:: AVAR;
|
3 .任何实现接口的类都要实现接口中所定义的全部方法
1
2
3
4
5
|
class
E
implements
iA
{
public
function
iAfunc1(){
echo
"in iAfunc1"
;}
public
function
iAfunc2(){
echo
"in iAfunc2"
;}
}
|
不然该类必须声明为 abstract 。
1
|
abstract
class
E
implements
iA{}
|
4 .一个类能够在声明中使用 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"
;}
}
|
5 .接口不能够实现另外一个接口,但能够继承多个
1
2
3
4
5
6
7
8
|
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"
;}
}
|
3、 抽象类和接口的异同
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) 规范了一组相互协调的方法,其中一些方法是共同的,与状态无关的,能够共享的,无需子类分别实现;而另外一些方法却须要各个子类根据本身特定的状态来实现特 定的功能 。