PHP中的魔術方法

概述

在面向對象編程中,PHP提供了一系列的魔術方法,這些魔術方法為編程提供了很多便利。PHP中的魔術方法通常以__(兩個下劃線)開始,并且不需要顯示的調用而是由某種特定的條件出發。這篇文章簡單總結了PHP中提供的魔術方法。

開始之前

在總結PHP的魔術方法之前先來定義兩個類,以便后邊示例使用:

復制代碼代碼如下:

<?php
class Device {
??? public $name;???????????
??? public $battery;????????
??? public $data = array();?
??? public $connection;?????
?
??? protected function connect() {
??????? $this->connection = 'resource';
??????? echo $this->name . ' connected' . PHP_EOL;
??? }
?
??? protected function disconnect() {
??????? $this->connection = null;
??????? echo $this->name . ' disconnected' . PHP_EOL;
??? }
}
?
class Battery {
??? private $charge = 0;
?
??? public function setCharge($charge) {
??????? $charge = (int)$charge;
??????? if($charge < 0) {
??????????? $charge = 0;
??????? }
??????? elseif($charge > 100) {
??????????? $charge = 100;
??????? }
??????? $this->charge = $charge;
??? }
}
?>

Device類有四個成員屬性和兩個成員方法。Battery類有一個成員屬性和一個成員方法。

構造函數和析構函數

構造函數和析構函數分別在對象創建和銷毀時被調用。對象被“銷毀”是指不存在任何對該對象的引用,比如引用該對象的變量被刪除(unset)、重新賦值或腳本執行結束,都會調用析構函數。

__construct()

__construct()構造函數是目前為止最經常使用的函數。在創建對象時,可以在構造函數中做一些初始化工作。可以為構造函數定義任意多個參數,只要在實例化時傳入對應個數的參數即可。構造函數中出現的任何異常都會阻止對象的創建。

復制代碼代碼如下:

class Device {
?? public function? __construct(Battery $battery, $name) {
?????? $this->battery = $battery;
?????? $this->name = $name;
?????? $this->connect();
??? }
}

上面的示例代碼中,Device類的構造函數為成員屬性賦值并且調用了connect()方法。

復制代碼代碼如下:

將構造函數聲明為私有方法,可以防止在類外部創建對象,這在單利模式中經常使用。

__desctruct()

析構函數通常在對象被銷毀時調用,析構函數不接收任何參數。經常在析構函數中執行一些清理工作,比如關閉數據庫連接等。

屬性重載(Property Overloading)

有一點需要注意的是:PHP中的”重載”與其他大多數語言的重載不是太一樣,雖然都實現了相同的功能。
屬性重載涉及到的兩個魔術方法主要是用來處理屬性訪問,定義了當我們嘗試訪問一個不存在(或不可訪問)的屬性時會發生什么。

__get()

魔術方法__get()在我們嘗試訪問一個不存在的屬性時會被調用。它接收一個參數,該參數表示訪問屬性的名字,并且將該屬性的值返回。在上面的Device類里,有一個data屬性,該屬性就在這里就起了作用,如下面得代碼:

復制代碼代碼如下:

class Device {
??? public function? __get($name) {
???????? if(array_key_exists($name, $this->data)) {
??????????? return $this->data[$name];
??????? }
??????? return null;
??? }
}

該魔術方法最常用的地方就是通過創建一個“只讀”的屬性來擴展訪問控制。在上面的Battery類中,有一個私有屬性$charge,我們可以通過__get()魔術方法將該屬性擴展為在類外部可讀但不能修改。代碼如下:

復制代碼代碼如下:

class Battery {
??? private $charge = 0;
?
??? public function? __get($name) {
??????? if(isset($this->$name)) {
??????????? return $this->$name;
??????? }
??????? return null;
??? }
}

__set()

__set()魔術方法在我們嘗試修改一個不可訪問的屬性時會被調用,它接收兩個參數,一個表示屬性的名字,一個表示屬性的值。示例代碼如下:

復制代碼代碼如下:

class Device {
??? public function? __set($name, $value) {
??????? // use the property name as the array key
??????? $this->data[$name] = $value;
??? }
}

__isset()

__isset()魔術方法在對一個不可訪問的屬性調用isset()方法時會被調用,它接收一個參數,表示屬性的名字。它應該返回一個布爾值,用來表示該屬性是否存在。代碼如下:

復制代碼代碼如下:

class Device {
??? public function? __isset($name) {
??????? return array_key_exists($name, $this->data);
??? }
}

__unset()

__unset()魔術方法在調用unset()函數銷毀一個不能訪問的屬性時會被調用,它接收一個參數,表述屬性的名字。

對象轉換為字符串

有時候我們需要將對象以字符串的形式表現出來。如果我們直接打印一個對象,那么程序將會輸出一個錯誤信息:PHP Catchable fatal error: Object of class Device could not be converted to string

__toString()

__toString()在我們將對象當作字符串一樣使用時會被調用,它不接收任何參數。該方法允許我們定義對象的表現形式。代碼如下:

復制代碼代碼如下:

class Device {
??? public function? __toString() {
?????? $connected = (isset($this->connection)) ? 'connected' : 'disconnected';
?????? $count = count($this->data);
?????? return $this->name . ' is ' . $connected . ' with ' . $count . ' items in memory' . PHP_EOL;
??? }
??? ...
}

__set_state()(PHP 5.1)

靜態魔術方法__set_state(),在我們使用var_export()函數輸出對象時會調用該方法。var_export()函數用來將PHP變量轉換為PHP代碼,它接收一個包含對象屬性值的關聯數組作為參數。示例代碼如下:

復制代碼代碼如下:

class Battery {
??? //...
??? public static function? __set_state(array $array) {
??????? $obj = new self();
??????? $obj->setCharge($array['charge']);
??????? return $obj;
??? }
??? //...
}

克隆對象

默認的,對象都是按引用傳值的。因此,在將一個對象賦值給另一個變量時,只是創建了指向該對象的一個引用,并沒有復制該對象。為了實現真正得復制一個對象,我們需要使用clone關鍵字。
這種“按引用傳遞”的策略同樣適用于包含在對象內部的對象。即使我們克隆了一個對象,在對象內部的任何對象都不會被克隆,因此最終的結果是兩個對象共享了同一個內部對象。示例代碼如下:

復制代碼代碼如下:

$device = new Device(new Battery(), 'iMagic');
$device2 = clone $device;
?
$device->battery->setCharge(65);
echo $device2->battery->charge;
// 65

__clone()

__clone()魔術方法__clone()可以解決上面的問題。當對一個對象使用clone關鍵字時,該魔術方法會被調用。在這個魔術方法里,我們可以實現任何子對象的克隆,代碼如下:

復制代碼代碼如下:

class Device {
??? ...
??? public function? __clone() {
??????? // copy our Battery object
??????? $this->battery = clone $this->battery;
??? }
??? ...
}

對象序列化

序列化是講任意數據轉換為字符串格式的過程。序列化通常用來將整個對象存入數據庫或寫入文件中。當反序列化存儲的數據時,我們可以得到序列化之前的對象。但是,并不是所有得數據都可以被序列化,比如數據庫連接。幸運的是,有一個魔術方法可以幫我們解決這個問題。

__sleep()

魔術方法__sleep()在對一個對象序列化時(調用serialize())會被調用。它不接收任何參數,而且應該返回一個包含所有應該被序列化的屬性的數組。在該魔術方法中,也可以執行一些其他操作。
有一點要注意的是,不要再該函數中進行任何的析構操作,因為這可能會影響正在運行的對象。

示例代碼如下:

復制代碼代碼如下:

class Device {
??? public $name;???????????
??? public $battery;???????
??? public $data = array();
??? public $connection;????
??? //...
??? public function? __sleep() {
??????? return array('name', 'battery', 'data');
??? }
??? //...
}

__wakeup()

魔術方法__wakeup()在對存儲的對象反序列化時會被調用。它不接收任何參數,也沒有任何返回值。可以用它來處理在序列化時丟失的數據庫連接或資源。代碼如下:

復制代碼代碼如下:

class Device {
??? //...
??? public function? __wakeup() {
??????? // reconnect to the network
??????? $this->connect();
??? }
??? //...
}

方法重載

PHP還有兩個與成員方法相關的魔術方法__call()和__callStatic(),這兩個魔術方法類似于屬性重載方法。

__call()

魔術方法__call()在調用不存在或不可訪問的方法時會被調用。它接收兩個參數,一個是調用的方法的名字,一個是包含函數參數的數組。我們可以使用這種方法調用子對象中得同名函數。

在這個例子中,要注意函數call_user_func_array(),這個函數允許我們動態調用一個命名的函數。

示例代碼如下:

復制代碼代碼如下:

class Device {
??? //...
??? public function? __call($name, $arguments) {
??????? // make sure our child object has this method
??????? if(method_exists($this->connection, $name)) {
??????????? // forward the call to our child object
??????????? return call_user_func_array(array($this->connection, $name), $arguments);
??????? }
??????? return null;
??? }
??? //...
}

__callStatic()

魔術方法__callStatic()與__call()的功能一樣,唯一不同的是,該方法在嘗試訪問一個不存在或不可訪問的靜態方法時會被調用。示例代碼如下:

復制代碼代碼如下:

class Device {
??? //...
??? public static function? __callStatic($name, $arguments) {
??????? // make sure our class has this method
??????? if(method_exists('Connection', $name)) {
??????????? // forward the static call to our class
??????????? return call_user_func_array(array('Connection', $name), $arguments);
??????? }
??????? return null;
??? }
??? //...
}

將對象作為函數

有時候我們會需要將對象作為函數使用。將對象作為函數使用,就像我們使用其他普通的函數一樣,可以傳參。

__invoke()(PHP 5.3)

魔術方法__invoke()在嘗試將對象作為函數使用時會被調用。在該方法中定義的任何參數,都將被作為函數的參數。示例代碼如下:

復制代碼代碼如下:

class Device {
??? //...
??? public function __invoke($data) {
??????? echo $data;
??? }
??? //...
}
$device = new Device(new Battery(), 'iMagic');
$device('test');
// equiv to $device->__invoke('test')
// Outputs: test

其他:__autoload()

__autoload()方法并不是一個魔術方法,但是這個方法非常有用。但是,對著PHP版本的更新,該函數已經不建議使用,取而代之的是spl_auto_register()函數。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/250595.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/250595.shtml
英文地址,請注明出處:http://en.pswp.cn/news/250595.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

執行caffe的draw_net.py出現“GraphViz's executable dot not found”的解決方法

執行caffe的draw_net.py出現“GraphVizs executable "dot" not found”的解決方法 控制臺輸入如下指令畫網絡圖&#xff1a;python ../../../python/draw_net.py train.prototxt train.png --rankdirTB &#xff08;Top-Bottom形式&#xff0c;縱向圖&#xff09;pyt…

配置 --- vscode自定義代碼段Snippets

目標 在vscode中輸入vbs-vue 然后產生一個自己想要的模板 寫好模板 在線上寫好模板傳送門: https://snippet-generator.app/ 1是標題,對應 2是前綴.對應在vue中使用的快捷鍵 vbs-vue3就是需要顯示的代碼段了 在vscode中配置 1.ctrlshiftp2.選擇 Preferences: Configure U…

centos6安裝composer

需要使用到curl&#xff0c;沒有的話需要 yum -y install curl ###安裝一、下載&#xff1a;curl -sS https://getcomposer.org/installer | php &#xff08;如果是網絡原因多試幾次&#xff09; 二、移動composer.phar移動到環境下讓其變成可執行&#xff1a;mv compose…

透明圖與元素居中

1,定位讓元素居中 1. 透明度 opacity 默認值是1 不透明 0是全透明轉載于:https://www.cnblogs.com/Shinigami/p/9709382.html

配置 --- vscode中react格式化解決方案

選擇右下角的語言 在彈出框搜react選擇 JavaScript React(或者根據需求選擇 TypeScript React) 快捷鍵, windows下 Alt SHIFT F

【商城購物車】購物車邏輯

轉載于:https://www.cnblogs.com/xuzhengzong/p/8746677.html

PHP遞歸實現無限極分類

PHP遞歸實現無限極分類 摘要 今天在編碼的時候要用到二級的欄目分類&#xff0c;所以順便就把無限極分類給整理了一下&#xff0c;采用的是遞歸方法 //實現無限級分類public function getTree(){$categorys Category::all();return $this->makeTree($categorys, cate_id,…

IO NIO

1,Java NIO Java non-blocking IO 即 非阻塞IO,線程在等待的時候&#xff0c;可以做其他的事情。 2,IO 對比NIO IO 是面向流&#xff0c;NIO 是面向緩沖 面向流是指每次從流中讀出一個或者多個字節&#xff0c;直到全部讀出為止 面向緩沖區是指將數據先存到一個緩存區 IO 是阻…

react --- 生命周期 給子組件傳遞數據

子組件 /src/components/LifeCycle.js import React, { Component } from reactexport class LifeCycle extends Component {constructor(props) {super(props);// 常用于初始化狀態(狀態初始化、屬性初始化)console.log("1.組件構建函數執行");}componentWillMoun…

Vue---mock.js 使用

mockjs 概述 在我們的生產實際中&#xff0c;后端的接口往往是較晚才會出來&#xff0c;并且還要寫接口文檔&#xff0c;于是我們的前端的許多開發都要等到接口給我們才能進行&#xff0c;這樣對于我們前端來說顯得十分的被動&#xff0c;于是有沒有可以制造假數據來模擬后端接…

Java 的抽象類

Java 的抽象類 用abstract關鍵字來修飾一個類時&#xff0c;這個類叫做抽象類&#xff1b;用abstract來修飾一個方法時&#xff0c;該方法叫做抽象方法。 抽象方法&#xff1a;只有方法的聲明&#xff0c;沒有方法的實現。以分號結束&#xff1a;abstract int abstractMethod…

react --- 按需加載組件

問題描述 使用 antd庫時使用按鈕,須導入如下 import Button from antd/lib/button import antd/dist/antd.css這樣會導入全局的樣式. 解決方案,配置按需加載 1.安裝 react-app-rewired取代 react-scripts, 可以擴展webapack 的配置, 類似vue.config.jsnpm install react-ap…

flask 實現異步非阻塞----gevent

我們都知道&#xff0c;flask不支持異步非阻塞的請求&#xff0c;我們可以創建一個新項目去測試一下&#xff0c;推薦大家使用pycharm去開發我們的flask 使用特別的方便。 rom flask import Flask import time app Flask(__name__) app.route(/) def hello_world():time.slee…

Axure下拉框級聯操作

現實生活中有很多的下拉框是級聯操作的&#xff0c;即因為第一個下拉框的選擇&#xff0c;影響到后面的下拉框的選擇的列表的數據。或許在代碼中&#xff0c;這些操作相對比較簡單&#xff0c;通過前一個下拉框的選擇項來控制后一個下拉框的數據的動態添加。那么&#xff0c;如…

react --- render持續調用解決方案

問題描述: 在某個組件中.有可能頻繁的取數據(但是數據未改變,因此不需要更新).數據的頻繁請求會觸發render函數,造成性能消耗模擬代碼如下 export class CommentList extends Component {constructor(props) {super(props);this.state {comments: []}}// 模擬頻繁的獲取新數…

Java 的工廠方法及代理模式

Java 的工廠方法及代理模式 工廠方法(FactoryMethod) 概述&#xff1a;定義一個用于創建對象的接口&#xff0c;讓子類決定實例化哪一個類。FactoryMethod使一個類的實例化延遲到其子類。 適用性&#xff1a; 當一個類不知道它所必須創建的對象的類的時候當一個類希望由它的…

Linux 和 Vim 常用命令整理

Sftp常用命令&#xff1a; lcd f&#xff1a;本地切換到 F盤 lpwd本地 當前目錄 lls本地 文件列表 put 本地 上傳文件到服務器(put輸入后&#xff0c;回車會有彈窗&#xff0c;選擇上傳文件) get下載文件到本地 Linux命令&#xff1a; 1.文件夾的操作 1 pwd&#xff1a;顯示…

Socket網絡編程——C++實現

本代碼可直接使用 根據TCP/IP三次握手&#xff0c;實驗時可使用兩臺電腦&#xff0c;或者打開兩個終端模擬通信。 服務器端&#xff1a; #include <iostream> #include <windows.h>using namespace std;#pragma comment(lib,"ws2_32.lib") //引用靜態鏈接…

react --- 復合組件,傳遞屬性

組件復合 復合組件給予你足夠的靈活去定義組件的外觀和行為,而且是以一種明確和安全的方式進行.如果組件間有公用的非UI邏輯,將它們抽取為JS模塊導入使用而不是繼承它/src/components/Composition.js // Dialog作為容器不關心內容和邏輯 function Dialog(props){return <d…

6、復制文件

復制文件 要求&#xff1a; 1、將原文件xxx.txt中的內容復制到新的文件里 2、新文件的文件名為xxx&#xff08;復制&#xff09;.txt&#xff0c;即原文件名復制進行命名 大框架&#xff1a; 1、輸入想要復制的文件xxx.txt input() 2、創建一個文件xxx(復制).txt f1 open(&quo…