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) 規範了一組相互協調的方法,其中一些方法是共同的,與狀態無關的,能夠共享的,無需子類分別實現;而另外一些方法卻須要各個子類根據本身特定的狀態來實現特 定的功能 。