PHP5對象體系_PHP

來源:互聯網
上載者:User
關鍵字 體系 對象 PHP5 類型 function 使用
* 本文是對《Classes and Objects in PHP5》系列文章的補充和修正,介紹了PHP5對象體系的總體架構,但有些特性沒有具體介紹。強烈建議在讀過《Classes and Objects in PHP5》後閱讀本文。



PHP5推出的對象體系相信是大家最為期待的。PHP5借鑒了Java2的物件模型,提供了較為強大的物件導向編程支援,使用PHP來實現OO將變得輕鬆和自然。



對象傳遞



PHP5使用了Zend引擎II,對象被儲存於獨立的結構Object Store中,而不像其它一般變數那樣儲存於Zval中(在PHP4中對象和一般變數一樣儲存於Zval)。在Zval中僅儲存物件的指標而不是內容(value)。當我們複製一個對象或者將一個對象當作參數傳遞給一個函數時,我們不需要複製資料。僅僅保持相同的對象指標並由另一個zval通知現在這個特定的對象指向的Object Store。由於對象本身位於Object Store,我們對它所作的任何改變將影響到所有持有該對象指標的zval結構----表現在程式中就是目標對象的任何改變都會影響到來源物件。.這使PHP對象看起來就像總是通過引用(reference)來傳遞,因此PHP中對象預設為通過“引用”傳遞,你不再需要像在PHP4中那樣使用&來聲明。



記憶體回收機制

某些語言,最典型的如C,需要你顯式地要求分配記憶體當你建立資料結構。一旦你分配到記憶體,就可以在變數中儲存資訊。同時你也需要在結束使用變數時釋放記憶體,這使機器可以空出記憶體給其它變數,避免耗光記憶體。

PHP可以自動進行記憶體管理,清除不再需要的對象。PHP使用了引用計數(reference counting)這種單純的記憶體回收(garbage collection)機制。每個對象都內含一個引用計數器,每個reference串連到對象,計數器加1。當reference離開生存空間或被設為NULL,計數器減1。當某個對象的引用計數器為零時,PHP知道你將不再需要使用這個對象,釋放其所佔的記憶體空間。

例如:

class Person{
}
function sendEmailTo(){
}

$haohappy = new Person( );
// 建立一個新對象: 引用計數 Reference count = 1
$haohappy2 = $haohappy;
// 通過引用複製: Reference count = 2
unset($haohappy);
// 刪除一個引用: Reference count = 1
sendEmailTo($haohappy2);
// 通過引用傳遞對象:
// 在函數執行期間:
// Reference count = 2
// 執行結束後:
// Reference count = 1

unset($haohappy2);
// 刪除引用: Reference count = 0 自動釋放記憶體空間

?>



以上是PHP5在記憶體管理上的變化,也許大家不怎麼感興趣。下面我們來看看PHP5中的物件模型和PHP4有什麼具體的不同之處:



★ 新增功能

★ 改進功能



1) ★ Private and Protected Members 私人和保護類成員(屬性,方法)

2) ★ Abstract Classes and Methods 抽象類別和抽象方法

3) ★ Interfaces             介面

4) ★ Class Type Hints  類型指示 =

5) ★ final  final關鍵字 =

6) ★ Objects Cloning  對象複製 =

7) ★ Constructors and Destructors 建構函式和解構函式

8) ★ Class Constants 類常量 =

9) ★ Exceptions 異常處理

10) ★ Static member 靜態類成員

11) ★__METHOD__ constant __METHOD__常量 =

12) ★ Reflection 反射機制 



第1、2、3、7、10請自行查閱本文末尾的《Classes and Objects in PHP5》系列,其中已有詳細介紹,本文中不再講解。第9點異常處理和第12點反射機制內容較為豐富,限於篇幅亦不在文中介紹,請關注即將推出的《PHP & More》電子雜誌第二期,會專門撰文介紹。



以下向大家介紹第4、5、6、8、11點語言特性:



4) ★ Class Type Hints  類型指示



大家都知道,PHP是一種弱類型的語言。在使用變數前不需要定義,不需要聲明變數的資料類型。這在編程中帶來很多便利,但也帶了一些隱患,特別當變數的類型變化時。在PHP5增加了類型指示,可以在執行過程中自動對類方法的參數類型進行判斷。這類似於Java2中的RTTI,配合reflection可以讓我們很好地控制對象。





interface Foo {
function a(Foo $foo);
}

interface Bar {
function b(Bar $bar);
}

class FooBar implements Foo, Bar {
function a(Foo $foo) {
// ...
}

function b(Bar $bar) {
// ...
}
}

$a = new FooBar;
$b = new FooBar;

$a->a($b);
$a->b($b);
?>



在強型別語言中,所有變數的類型將在編譯時間進行檢查,而在PHP中使用類型指示來對類型的檢查則發生在運行時。如果類方法參數的類型不對,將會報出類似“Fatal error: Argument 1 must implement interface Bar…”這樣的錯誤資訊。



以下代碼:

function foo(ClassName $object) {
// ...
}
?>



相當於:

function foo($object) {
if (!($object instanceof ClassName)) {
die("Argument 1 must be an instance of ClassName");
}
}
?>





5) ★ final  final關鍵字



PHP5中新增加了final關鍵字,它可以加在類或類方法前。標識為final的類方法,在子類中不能被覆寫。標識為final的類,不能被繼承,而且其中的方法都預設為final類型。

Final方法:

class Foo {
final function bar() {
// ...
}
}
?>



Final類:

final class Foo {
// class definition
}

// 下面這一行是錯誤的
// class Bork extends Foo {}
?>



6) ★ Objects Cloning  對象複製

前面在記憶體管理部份說過,PHP5中預設通過引用傳遞對象。像使用$object2=$object1這樣的方法複製出的對象是相互關聯的。如果我們確實需要複製出一個值與原來相同的對象而希望目標對象與來源物件沒有關聯(像普通變數那樣通過值來傳遞),那麼就需要使用clone關鍵字。如果還希望在複製的同時變動來源物件中的某些部份,可以在類中定一個__clone()函數,加入操作。



//對象複製
class MyCloneable {
static $id = 0;

function MyCloneable() {
$this->id = self::$id++;
}


/*
function __clone() {
$this->address = "New York";
$this->id = self::$id++;
}
*/
}

$obj = new MyCloneable();

$obj->name = "Hello";
$obj->address = "Tel-Aviv";

print $obj->id . "\n";

$obj_cloned = clone $obj;

print $obj_cloned->id . "\n";
print $obj_cloned->name . "\n";
print $obj_cloned->address . "\n";
?>



以上代碼複製出一個完全相同的對象。



然後請把function __clone()這一個函數的注釋去掉,重新運行程式。則會複製出一個基本相同,但部份屬性變動的對象。



8) ★ Class Constants 類常量

PHP5中可以使用const關鍵字來定義類常量。



class Foo {
const constant = "constant";
}

echo "Foo::constant = " . Foo::constant . "\n";
?>

















11) ★__METHOD__ constant __METHOD__常量

__METHOD__ 是PHP5中新增的“魔術”常量,表示類方法的名稱。
魔術常量是一種PHP預定義常量,它的值可以是變化的,PHP中的其它已經存在的魔術常量有__LINE__、__FILE__、__FUNCTION__、__CLASS__等。

class Foo {
function show() {
echo __METHOD__;
}
}

class Bar extends Foo {
}

Foo::show(); // outputs Foo::show
Bar::show(); // outputs Foo::show either since __METHOD__ is
// compile-time evaluated token

function test() {
echo __METHOD__;
}

test(); // outputs test
?>
  • 相關文章

    聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.