抽象方法和抽象類
在OOP語言中,一個類可以有一個或多個子類,而每個類都有至少一個公有方法做為外部代碼訪問其的接口。而抽象方法就是為了方便繼承而引入的,我們先來看一下抽象類和抽象方法的定義再說明它的用途。
什麼是抽象方法?我們在類裡面定義的沒有方法體的方法就是抽象方法,所謂的沒有方法體指的是,在方法聲明的時候沒有大括號以及其中的內容,而是直接在聲明時在方法名後加上分號結束,另外在聲明抽象方法時還要加一個關鍵字“abstract”來修飾;例如:
代碼如下 復制代碼abstract function fun1();
abstract function fun2();
上例是就是“abstract”修飾的沒有方法體的抽象方法“fun1()”和“fun2()”,不要忘記抽象方法後面還要有一個分號;那麼什麼是抽象類呢?只要一個類裡面有一個方法是抽象方法,那麼這個類就要定義為抽象類,抽象類也要使用“abstract”關鍵字來修飾;在抽象類裡面可以有不是抽象的方法和成員屬性,但只要有一個方法是抽象的方法,這個類就必須聲明為抽象類,使用”abstract”來修飾。例如:
代碼如下 復制代碼abstract class Demo
{
var $test;
abstract function fun1();
abstract function fun2();
function fun3()
{
... ...
}
}
上例中定義了一個抽象類“Demo”使用了”abstract”來修飾, 在這個類裡面定義了一個成員屬性“$test”,和兩個抽象方法“fun1”和“fun2”還有一個非抽象的方法fun3();那麼抽象類我們怎麼使用呢?最重要的一點就是抽象類不能產生實例對象,所以也不能直接使用,前面我們多次提到過類不能直接使用,我們使用的是通過類實例化出來的對象,那麼抽象類不能產生實例對象我們聲明抽象類有什麼用呢?我們是將抽象方法是做為子類重載的模板使用的,定義抽象類就相當於定義了一種規范,這種規范要求子類去遵守,子類繼函抽象類之後,把抽象類裡面的抽象方法按照子類的需要實現。子類必須把父類中的抽象方法全部都實現,否則子類中還存在抽象方法,那麼子類還是抽象類,還是不能實例化對;為什麼我們非要從抽象類中繼承呢?因為有的時候我們要實現一些功能就必須從抽象類中繼承,否則這些功能你就實現不了,如果繼承了抽象類,就要實現類其中的抽象方法;
代碼如下 復制代碼abstract class Demo
{
var $test;
abstract function fun1();
abstract function fun2();
function fun3()
{
... ...
}
}
//抽象類為能產生實例對象,所以這樣做是錯的,實例化對象交給子類
代碼如下 復制代碼$demo=new Demo();
class Test extends Demo
{
function fun1()
{
... ...
}
function fun2()
{
... ...
}
}
//子類可以實例化對象,因為實現了父類中所有抽象方法
代碼如下 復制代碼 $test=new Test();__call處理調用錯誤
在程序開發中,如果在使用對象調用對象內部方法時候,調用的這個方法不存在那麼程序就會出錯,然後程序退出不能繼續執行。那麼可不可以在程序調用對象內部不存在的方法時,提示我們調用的方法及使用的參數不存在,但程序還可以繼續執行,這個時候我們就要使用在調用不存在的方法時自動調用的方法”__call()”。
//這是一個測試的類,裡面沒有屬性和方法
class Test
{
}
//產生一個Test類的對象
$test=new Test();
//調用對象裡不存在的方法
$test->demo("one", "two", "three");
//程序不會執行到這裡
echo "this is a test";
上例出現如下錯誤,程序通出不能繼續執行;
Fatal error: Call to undefined method Test::demo()
下面我們加上“__call()”方法,這個方法有2個參數,第一個參數為調用不存在的方法過程中,自動調用__call()方法時,把這個不存在的方法的方法名傳給第一個參數,第二個參數則是把這個方法的多個參數以數組的形式傳進來。
代碼如下 復制代碼//這是一個測試的類,裡面沒有屬性和方法
class Test
{
//調用不存的方法時自動調用的方法,第一個參數為方法名,第二個參數是數組參數
function __call($function_name, $args)
{
print "你所調用的函數:$function_name(參數:";
print_r($args);
print ")不存在!n";
}
}
//產生一個Test類的對象
$test=new Test();
//調用對象裡不存在的方法
$test->demo("one", "two", "three");
//程序不會退出可以執行到這裡
echo "this is a test";
上例輸出結果為:
你所調用的函數: demo(參數:Array ( [0] => one [1] => two [2] => three ) )不存在! this is a test.
克隆對象
有的時候我們需要在一個項目裡面,使用兩個或多個一樣的對象,如果你使用“new”關鍵字重新創建對象的話,再賦值上相同的屬性,這樣做比較煩瑣而且也容易出錯,所以要根據一個對象完全克隆出一個一模一樣的對象,是非常有必要的,而且克隆以後,兩個對象互不干擾。
在PHP5中我們使用”clone”這個關鍵字克隆對象;
代碼如下 復制代碼class Person
{
//下面是人的成員屬性
var $name; //人的名子
var $sex; //人的性別
var $age; //人的年齡
//定義一個構造方法參數為屬性姓名$name、性別$sex和年齡$age進行賦值
function __construct($name="", $sex="", $age="")
{
$this->name=$name;
$this->sex=$sex;
$this->age=$age;
}
//這個人可以說話的方法, 說出自己的屬性
function say()
{
echo "我的名子叫:".$this->name." 性別:".$this->sex." 我的年齡是:".$this->age."";
}
}
$p1=new Person("張三", "男", 20);
//使用“clone”克隆新對象p2,和p1對象具有相同的屬性和方法。
$p2=clone $p1;
$p2->say();
PHP5定義了一個特殊的方法名“__clone()”方法,是在對象克隆時自動調用的方法,用“__clone()”方法將建立一個與原對象擁有相同屬性和方法的對象,如果想在克隆後改變原對象的內容,需要在__clone()中重寫原本的屬性和方法, ”__clone()”方法可以沒有參數,它自動包含$this和$that兩個指針,$this指向復本,而$that指向原本;
代碼如下 復制代碼class Person
{
//下面是人的成員屬性
var $name; //人的名子
var $sex; //人的性別
var $age; //人的年齡
//定義一個構造方法參數為屬性姓名$name、性別$sex和年齡$age進行賦值
function __construct($name="", $sex="", $age="")
{
$this->name=$name;
$this->sex=$sex;
$this->age=$age;
}
//這個人可以說話的方法, 說出自己的屬性
function say()
{
echo "我的名子叫:".$this->name." 性別:".$this->sex." 我的年齡是:".$this->age."";
}
//對象克隆時自動調用的方法, 如果想在克隆後改變原對象的內容,需要在__clone()中重寫原本的屬性和方法
function __clone()
{
//$this指的復本p2, 而$that是指向原本p1,這樣就在本方法裡,改變了復本的屬性。
$this->name="我是假的$that->name";
$this->age=30;
}
}
$p1=new Person("張三", "男", 20);
$p2=clone $p1;
$p1->say();
$p2->say();
上例輸出:
我的名子叫:張三 性別:男 我的年齡是:20
我的名子叫:我是假的張三 性別:男 我的年齡是:30