PHP中 對象自動調用的方法:set()、get()、tostring()

來源:互聯網
上載者:User
總結:

(1)__get($property_name):擷取私人屬性$name值時,此對象會自動調用該方法,將屬性name值傳給參數$property_name,通過這個方法的內部

執行,返回我們傳入 的私人屬性的值。

(2)__set($property_name, $value):直接給私人屬性賦值時,此對象會自動調用該方法,把屬性比如name傳給$property_name, 把要賦的值

“zhangsan”傳給$value,通過這個方法的執行,達到賦值的目的。

(3)__tostring() : 當直接輸出控制代碼(可以理解為一個執行個體)時,會自動執行__tostring()方法。

1.__set()與__get().

一般來說,總是把類的屬性定義為private,這更符合現實的邏輯。但是,對屬性的讀取和賦值操作是非常頻繁的,因此在 PHP5中,預定義了兩個函數

“__get()”和“__set()”來擷取和賦值其屬性,以及檢查屬性的“__isset()”和刪除屬性的方法 “__unset()”。


我們為每個屬性做了設定和擷取的方法,在PHP5中給我們提供了專門為屬性設定值和獲 取值的方法,“__set()”和“__get()”這兩個方法,這兩個方法不是預設存在的,

而是我們手工添加到類裡面去的,像構造方法 (__construct())一樣, 類裡面添加了才會存在,可以按下面的方式來添加這兩個方法,當然也可以按個人的風格來添加:

//__get()方法用來擷取私人屬性public function __get($property_name){  if(isset($this->$property_name)){ return($this->$property_name); }else{ return(NULL); }}//__set()方法用來設定私人屬性public function __set($property_name, $value){$this->$property_name = $value;}

__get()方法:這個方法用來擷取私人成員屬性值的,有一個參數,參數傳入 你要擷取的成員屬性的名稱,返回擷取的屬性值,這個方法不用我們手工的去調用,因為我們也可以把這個方法做成私人的方法,是在直接擷取私人屬性的時候對象 自動調用的。因為私人屬性已經被封裝上了,是不能直接擷取值的(比如:“echo $p1->name”這樣直接擷取是錯誤的),但是如果你在類裡面加上了這個方法,在使用“echo $p1->name”這樣的語句直接擷取值的時候就會自動調用__get($property_name)方法,將屬性name傳給參 數$property_name,通過這個方法的內部執行,返回我們傳入的私人屬性的值。如果成員屬性不封裝成私人的,對象本身就不會去自動調用這個方 法。


__set()方法:這個方法用來為私人成員屬性設定值的,有兩個參數,第一個參數為你要為設定值的屬性名稱,第二個參數是要給屬性設定的值, 沒有傳回值。這個方法同樣不用我們手工去調用,它也可以做成私人的,是在直接設定私人屬性值的時候自動調用的,同樣屬性私人的已經被封裝上

了, 如果沒有__set()這個方法,是不允許的,比如:$this->name=‘zhangsan’, 這樣會出錯,但是如果你在類裡面加上了__set($property_name, $value)這個方法,在直接給私人屬性賦值的時候,就會自動調用它,把屬性比如name傳給$property_name, 把要賦的值“zhangsan”傳給$value,通過這個方法的執行,達到賦值的目的。如果成員屬性不封裝成私人的,對象本身就不會去自動調用這個方 法。為了不傳入非法的值,還可以在這個方法給做一下判斷。代碼如下:

"; if (isset ( $this->$property_name )) { return ($this->$property_name); } else { return (NULL); } }  //__set()方法用來設定私人屬性 public function __set($property_name, $value) { echo "在直接設定私人屬性值的時候,自動調用了這個__set()方法為私人屬性賦值
"; $this->$property_name = $value; }}$per=new person();$per->name="shirayner"; //此時$per自動調用__set($property_name, $value)方法echo $per->name; //此時$per自動調用__get($property_name)方法?>2.__tostring() TOstring(在這裡故意這麼寫,是要說明PHP中方法不區分大小寫,但實際開發中還需要注意規範)。當進行測試時,需要知道是否得出正確的資料。比如列印一個對象時,看看這個對象都有哪些屬性,其值是什麼,如果類定義了toString方法,就能在測試時,echo列印對象體,對象就會自動調用它所屬類定義的toString方法,格式化輸出這個對象所包含的資料。如果沒有這個方法,那麼echo一個對象將報錯,例如“Catchable fatal error:Object of class Account could not be converted tostring”語法錯誤,實際上這是一個類型匹配失敗錯誤。不過仍然可以用print_r()和var_dump()函數輸出一個對象。當然,toString是可以定製的,所提供的資訊和樣式更豐富。
   
class Account{
public $user=1;
private $pwd=2;
// 自訂的格式化輸出方法
public function toString(){
return "當前對象的使用者名稱是{$this->user},密碼是{$this->pwd}";
}
}
$a=new Account();
echo $a;
echo PHP_EOL;
print_r($a);

運行這段代碼發現,使用toString方法後,輸出的結果是可定製的,更易於理解。實際上,PHP的toString魔術方法的設計原型來源於 Java。Java中也有這麼一個方法,而且在Java中,這個方法被大量使用,對於偵錯工具比較方便。實際上,toString方法也是一種序列化,我 們知道PHP內建serialize/unserialize也是進行序列化的,但是這組函數序列化時會產生一些無用資訊,如屬性字串長度,造成儲存空 間的無謂浪費。因此,可以實現自己的序列化和還原序列化方法,或者json_encode/json_decode也是一個不錯的選擇。

為什麼直接echo一個對象就會報語法錯誤,而如果這個對象實現toString方法後就可以直接輸出呢?原因很簡單,echo本來可以列印一個對 象,而且也實現了這個介面,但是PHP對其做了個限制,只有實現toString後才允許使用。從下面的PHP原始碼裡可以得到驗證:

ZEND_VM_HANDLER(40, ZEND_ECHO, CONST|TMP|VAR|CV, ANY)
{
zend_op *opline = EX(opline);
zend_free_op free_op1;
zval z_copy;
zval *z = GET_OP1_ZVAL_PTR(BP_VAR_R);
// 此處的代碼預留了把對象轉換為字串的介面
if (OP1_TYPE != IS_CONST &&
Z_TYPE_P(z) == IS_OBJECT && Z_OBJ_HT_P(z)->get_method != NULL &&
zend_std_cast_object_tostring(z, &z_copy, IS_STRING TSRMLS_CC) == SUCCESS) {
zend_print_variable(&z_copy);
zval_dtor(&z_copy);
} else {
zend_print_variable(z);
}
 
FREE_OP1();
ZEND_VM_NEXT_OPCODE();
}
  • 相關文章

    聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.