一、抽像類(abstract)
在我們實(shí)際開發(fā)過程中,有些類并不需要被實(shí)例化,如前面學(xué)習(xí)到的一些父類,主要是讓子類來繼承,這樣可以提高代碼復(fù)用性
語法結(jié)構(gòu):
復(fù)制代碼代碼如下:
abstract class 類名{
屬性 $name;
方法(){} //方法也可以為abstract 修飾符 function 方法名(){}
}
例:
復(fù)制代碼代碼如下:
abstract class animal{
public $name;
public $age;
//抽象方法不能有方法體,主要是為了讓子類去實(shí)現(xiàn);
abstract public function cry();
//抽象類中可以包含抽象方法,同時(shí)也可以包含實(shí)例類方法
public function getname(){
echo $this->name;
}
}
class Cat{
public function cry(){
echo 'ok';
}
}
理解:動(dòng)物類,實(shí)際上是一個(gè)抽象的概念,它規(guī)定了一些動(dòng)物有些哪共同的屬性和行為,但實(shí)際上它自己并沒收有那些屬性和行為。再比如:交通工具類,植物類等等
注意:
1、如果一個(gè)類用了abstract來修飾,則該類就是一個(gè)抽象類,如果一個(gè)方法被abstract修飾,那么該方法就是一個(gè)抽象方法,抽象方法不能有方法體=> abstract function cry(); 連{}也不可以有
2、抽象類一定不能被實(shí)例化,抽象類可以沒有抽象方法,但是如果一個(gè)類包含了任意一個(gè)抽象方法,這個(gè)類一定要聲明為abstract類;
3、如果一個(gè)類繼承了另一個(gè)抽象類,則該子類必須實(shí)現(xiàn)抽象類中所有的抽象方法(除非它自己也聲明為抽象類);
二、接口(interface)
接口就是將一些沒有實(shí)現(xiàn)的方法,封裝在一起,到某個(gè)類要用的時(shí)候,再根據(jù)具體情況把這些方法寫出來;
語法結(jié)構(gòu)
interface 接口名{
//屬性、方法
//接口中的方法都不能有方法體;
}
如何實(shí)現(xiàn)接口
class 類名 implements 接口名{
}
理解:接口就是更加抽象的抽象類,抽象類里的方法可以有方法體,但是接口中的方法必須沒有方法體。接口實(shí)現(xiàn)了程序設(shè)計(jì)的多態(tài)和高內(nèi)聚、低偶合的設(shè)計(jì)思想;
例:
復(fù)制代碼代碼如下:
//接口是定義規(guī)范、屬性的,一般以小寫的i開頭;
interface iUsb{
public function start();
public function stop();
}
//編寫相機(jī)類,讓它去實(shí)現(xiàn)接口
//當(dāng)一個(gè)類實(shí)現(xiàn)了某個(gè)接口,那么該類就必須實(shí)現(xiàn)接口的所有方法
class Camera implements iUsb{
public function start(){
echo 'Camera Start Work';
}
public function stop(){
echo 'Camera Stop Work';
}
}
//編寫一個(gè)手機(jī)類
class Phone implements iUsb{
public function start(){
echo 'Phone Satrt Work';
}
public function stop(){
echo 'Phone Stop Work';
}
}
$c=new Camera();
$c->start();
$p=new Phone();
$p->start();
什么時(shí)候使用接口:
1、定下規(guī)范,讓其他程序員來實(shí)現(xiàn)
2、當(dāng)多個(gè)平級(jí)的類,都需要去實(shí)現(xiàn)某個(gè)功能,但是實(shí)現(xiàn)的方式不一樣;
小結(jié):
1、接口不能被實(shí)例化,接口中所有的方法都不能有主體;
2、一個(gè)類可以實(shí)現(xiàn)多個(gè)接口,以逗號(hào)(,)分隔 class demo implements if1,if2,if3{}
3、接口中可以有屬性,但必須是常量,常量不可以有修飾符(默認(rèn)是public修飾符)
如:interface iUsb{
const A=90;
}
echo iUsb::A;
4、接口中的方法都必須是public的,默認(rèn)是public;
5、一個(gè)接口不能繼承其他的類,但是可以繼承其它的接口,一個(gè)接口可以繼承多個(gè)其它接口
如:interface 接口名 extends if1,if2{}
6、一個(gè)類可以在繼承父類的同時(shí)實(shí)現(xiàn)其它接口
如:class test extends testbase implements test1,test2{}
實(shí)現(xiàn)接口VS繼承類
php的繼承是單一繼承,也就是一個(gè)類只能繼承一個(gè)父類,這樣對(duì)子類功能的擴(kuò)展有一定的影響。實(shí)現(xiàn)接口可以看做是對(duì)繼承類的一個(gè)補(bǔ)充。繼承是層級(jí)的關(guān)系,不太靈活,而實(shí)現(xiàn)接口是平級(jí)的關(guān)系,實(shí)現(xiàn)接口可以在不打破繼承關(guān)系的前提下,對(duì)某個(gè)功能擴(kuò)展,非常靈活。
三、Final
1、如果我們希望某個(gè)類不被其它的類繼承(比如為了安全原因等。。),那么可以考慮使用final
語法:
final class A{}
2、如果我們希望某個(gè)方法,不被子類重寫,可以考慮使用final來修飾,final修飾的方法還是可以繼承的,因?yàn)榉椒ǖ睦^承權(quán)取決于public的修飾
如:
復(fù)制代碼代碼如下:
class A{
final public function getrate($salary){
return $salary*0.08;
}
}
class B extens A{
//這里父類的getrate方法使用了final,所以這里無法再重寫getrate
//public function getrate($salary){
// return $salary*0.01;
//}
}
3、final不能用來修飾屬性
四、類常量(const)
在某些情況下,可能有這樣的需求:當(dāng)不希望一個(gè)成員變量被修改,希望該變量的值是固定不變的,這時(shí)可以使用const常量(常量名應(yīng)該全用大寫,并且不帶$符號(hào),常量不可以加修飾符)
語法:
const 常量名=常量值; //必須賦初值,因?yàn)槌A渴遣荒苄薷牡?nbsp;
調(diào)用:
類名::常量名[本類內(nèi)部可用self::常量名] 或者 接口名::常量名 //接口中只能使用常量,不能使用變量
如:
復(fù)制代碼代碼如下:
class A{
const TAX_RATE=0.08;
function paytax($salary){
return $salary*self::TAX_RATE;
}
}
$a=new A();
echo $a->paytax(100);
注:
1、常量可以被子類繼承
2、常量是屬于某個(gè)類的,而不屬于某個(gè)對(duì)象