搜尋
首頁後端開發php教程PHP魔法步骤/函数详解

PHP魔法方法/函数详解
引自:
http://blog.csdn.net/inqihoo/article/details/9235103

在php的语法中,有一些系统自带的方法名,均以双下划线开头,它会在特定的情况下被调用。即所谓的魔法函数。
只有当你已经为一个给定类定义了这些魔法方法时,它们才会起作用。还要注意,这些方法不能直接调用,而只能通过要截获的事件来调用。
它们在面向对向编程中起着至关重要的作用。所以个人觉得很有必要整理一下。下面就php的15个魔法函数做一个详细整理:

1. __construct() 和 __destruct()
__construct() 对象初始化时会调用此方法(对于内核而言是指初始化完成后调用此方法).此方法用得比较多。__destruct() 当对象销毁时会调用此方法.那么什么时候对象会销毁呢?一是用户主动销毁对象,二是当程序结束时由引擎自动销毁
[php] view plaincopy

class People {        protected $_alive;        public function __construct()      {          $this->_alive = true;           echo 'Birth'.PHP_EOL;      }        public function __destruct()      {          $this->_alive = false;          echo 'Die'.PHP_EOL;      }  }    //主动销毁对象  $test = new Test();  unset($test);  sleep(1);    //程序完成后自动销毁对象  $test = new Test();  sleep(1);  


2. __get() 和 __set()
面向对象编程中使用频率很高的两个方法.当设置和获取对象的属性不允许访问时性,此方法会被调用。一定注意是不存在或不允许被读写时才会被调用。
因此对于一个对象,其属性不确定时,用这两个方法效果很好。

__get($name) 获取对象不存在的属性或无法访问的属性时调用.$name表示要获取的属性名
__set($name, $value) 设置对象不存在的属性或无法访问的属性时调用.$name表示要设置的属性名,$value表示要设置的值.

//例如:我们可以构建一个不确定属性的数据记录类  class Record {        protected $_data;         public function __get($name)      {          if (isset($this->_data[$name])) {              return $this->_data;          }          return false;      }        public function __set($name, $value)       {          $this->_data = $value;      }  }    $record = new Record();  $record->name = 'andrew';  echo 'My name is '.$record->name.PHP_EOL;  


3. __isset() 和 __unset()
这两个方法用得比较少些。当调用方法isset()判断对象是否存在某属性, 调用unset()注销某属性时。且当这些属性不存在或不可访问时,会分别调用__isset()和__unset()方法
与前面的__get()和__set()略同。都是某属性不存在或不可访问时被调用

__isset($name) 当调用方法isset()方法判断不可访问的类属性时调用.$name表示属性名.
__unset($name) 当调用方法unset()方法删除不可访问的类属性时调用.$name表示属性名.
//例如:  class People {        public $name;        public $sex;        private $_age;        public function __construct($name, $sex, $age)      {          $this->name = $name;          $this->sex  = $sex;          $this->_age = $age;      }        public function __isset($name)      {          echo 'The property '.$name.' not exists'.PHP_EOL;      }        public function __unset($name)      {          echo 'The property '.$name.' can not be unset'.PHP_EOL;      }  }    $people = new People('andrew', 'male', 28);  isset($people->name);  isset($people->real_name);  unset($people->_age);  


4. __call() 和 __callStatic()
前面,我们发现,在获取对象的属性时,如果此属性不存在会调用__get()方法。那么如果调用此对象的方法时,如果此方法不存在呢?php引擎会自动调用__call()方法。
同样,如果调用的是静态方法,且不存在时会调用__callStatic()方法。需要注意的是,__callStatic()使用时是一个静态方法,且仅在php5.3以上版本才支持.

__call($method, $args) 调用对象方法不存在或不允许被调用时此方法会被调用。$method表示调用的方法名,$args表示调用的参数
__callStatic($method, $args) 调用对象的静态方法不存在或不允许被调用时此方法会被调用。$method表示调用的方法名,$args表示调用的参数.
[php] view plaincopy
//例如:我们假定人只有跳走两种行为  class People {        public function jump()      {          echo 'I can jump'.PHP_EOL;      }        public function walk()      {          echo 'I can walk'.PHP_EOL;      }        public function __call($method, $args)      {          echo 'I can not '.$method.PHP_EOL;      }        //此方法必须是静态方法,且在php5.3版本下可用      public static function __callStatic($method, $args)      {          echo 'We can not'.$method.PHP_EOL;      }  }    $people = new People();  $people->jump();  $people->walk();  $people->fly();  People::fly();  


5. __sleep() 和 __wakeup()
这两个方法,咋一看,就是睡觉和唤醒嘛。那跟对象有什么关系?有时候该用的时候也想不起来。其实,我们简单点记,在php中有一个让对象睡觉的方法,叫searialize(),
它会将对象的各属性序列化以方便保存起来。而unsearialize()方法是将保存的序列化的数据解开变成对象。也叫唤醒。相对应的,当睡觉时,php会调用__sleep()方法,它
的返回值必须是一个数组,表示需要保存的属性项, 对于文件句柄,数据库连接等资源类型的数据是不能被序列化保存的。同理唤醒对象时,php会调用__wakeup()方法,
但与__sleep()不同的是,它返回值为空。被保存的属性都会被解开。那它有什么用呢?刚我们说了,searialize是不能保存资源的。那么唤醒时如果我们想用到这些资源怎么
办?回答很肯定,重新创建?那在哪里创建合适呢?当然是__wakeup()方法里面,因为每次唤醒时都会调用此方法嘛。这下我们很清楚这两个方法的用途了。

__sleep() 当调用searialize()方法时调用,返回值为数组,表示需要序列化的数据项.
__wakeup() 当调用unsearizlie()方法时调用。一般用来在唤醒时初始化资源对象.
[php] view plaincopy
//例如我们有一个用户类,用户名和性别都是类属性。用户的密码存在文件中  Class User {        public $username;        public $sex;        public $passFile;        private $_password;        public function __construct($username, $sex, $passFile)      {          $this->username = $username;          $this->sex      = $sex;          $this->passFile = $passFile;          $this->_password = file_get_contents($passFile);      }        public function getPassword()      {          return $this->_password;      }        public function __sleep()      {          return array(              'username', 'sex', 'passFile',          );      }        public function __wakeup()      {          $this->_password = file_get_contents($this->passFile);      }  }    $user = new User('andrew', 'male', 'pass.data');  $serializeData =  serialize($user);  echo $serializeData.PHP_EOL;  $user = unserialize($serializeData);  echo $user->getPassword().PHP_EOL;  


6. __toString()
当对象在需要转换成字符串时,会调用此方法。例如,echo对象时,将对象强制转换为string类型时, 用于字符串参数的函数中.注意:此方法的返回值必须为字符串。
[php] view plaincopy
//例如:  class Info {        public function __toString()      {          return "info";      }  }    $info = new Info();  echo $info.PHP_EOL;  echo md5($info).PHP_EOL;  echo (String) $info.PHP_EOL;  echo substr($info, 0, 2).PHP_EOL;  


7. __clone()
此方法在复制对象时被调用。我们知道在php中.$a为一个对象,$b=$a时。$b为$a的引用。当$a发生改变时。$b也会随之发生变化。那么为了使$b不发生变化,我们需要用$b=clone $a;
那么,当$a在调用clone的时候,引擎会自动调用__clone()方法
[php] view plaincopy
//例如:以下一个简单的例子    class Data {        public $value;        public function __clone()      {          echo "Clone myself".PHP_EOL;      }  }    $data = new Data();  $data->value = 4;  $newData = clone $data;  $data->value = 5;  echo "The data value is: ".$data->value.PHP_EOL;  echo "The new data value is: ".$newData->value.PHP_EOL;    //对面向对象比较熟的同学,一定对单例模式不陌生。PHP做单例模式的时候要记住把clone方法给禁掉。因为在单例模式中,是不允许复制对象的。如下例    class OnlyOne {        //单例对象      private static $_instance;        //不允许外部和子类调用初始化方法      private function __construct()      {}        //不允许复制      public function __clone()      {          throw new Exception('Not allow to clone me');      }        //获取单例对象      public static function getInstance()      {          if (self::$_instance != null) {              return self::$_instance;          }          self::$_instance = new self();          return self::$_instance;      }  }    $onlyOne = OnlyOne::getInstance();  $newOne  = clone $onlyOne;  


8.  __autoload()
autoload顾名思义就是自动加载。它主要用来自动加载类。那如何自动加载呢?我们都知道在php中,要使用另外一个文件中的类需要用require或include方法
(包括require_once和include_one)导入进来。那么如果我要使用的类未被导入,则引擎会自动调用__autoload()方法。利用此特性,当我们的类名和类文件有规律
地存放时,我们可以使用__autoload()方法,根据需导入的类名,让程序自动导入文件。此函数在许多的MVC框架中起着重要的作用。
__autoload($name) $name表示需要自动导入的类名
[php] view plaincopy
//我们制订规则类名以目录名加下载线拼接而成。例如:类Model_Config_Xml表示Model/Config/Normal.php文件.我们如何实现自动加载  //有类文件Model/Config/Normal.php内包含类如下:  class Model_Config_Normal {        public function __construct()      {          echo "Init Model_Config_Normal".PHP_EOL;      }  }      function __autoload($name)  {      $classPath = str_replace('_', DIRECTORY_SEPARATOR, $name);      require_once("$classPath.php");  }    //此时将无需再require('Model/Config/Normal.php')  $config = new Model_Config_Normal();  


9. __set_state()
这个方法用得不多。了解这个方法前,需要先知道var_export()函数,var_export()和var_dump()类似,输出一个变量的字符串表示。他与var_dump的区别在于它的返回结果的是合法的
php代码.此代码可以被eval执行. 注意:此方法是一个静态方法,且在php5.1以上版本才支持。
[php] view plaincopy
//例如:  class Test {        public $name;        public $age;        public static function __set_state($data)      {          $obj = new Test();          $obj->name = $data['name'];          $obj->age  = $data['age'] + 1;          return $obj;      }  }    $test = new Test();  $test->name = 'andrew';  $test->age  = 27;  $code =var_export($test, true);  echo $code.PHP_EOL;  eval('$new='.$code.';');  var_dump($new);  


10. __invoke()
这个方法我刚开始接触的时候,一眼望去,真看不出来它到底干嘛的。主要是不明白invoke是啥意思。后来查了字典才明白。意思是呼叫。英语差伤不起呀。
在php中这个方法用于,把对象当方法用的时候。此方法会被调用。很简单。注意,此方法仅5.3以上版本支持。
[php] view plaincopy
class Invoke {        public function __invoke()      {          echo 'I can run'.PHP_EOL;      }  }    $invoke = new Invoke();  $invoke();  

陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
您如何在PHP中創建和使用接口?您如何在PHP中創建和使用接口?Apr 30, 2025 pm 03:40 PM

本文解釋瞭如何創建,實施和使用PHP中的接口,重點關注其對代碼組織和可維護性的好處。

crypt()和password_hash()有什麼區別?crypt()和password_hash()有什麼區別?Apr 30, 2025 pm 03:39 PM

本文討論了PHP中的crypt()和password_hash()的差異,以進行密碼哈希,重點介紹其實施,安全性和對現代Web應用程序的適用性。

如何防止PHP中的跨站點腳本(XSS)?如何防止PHP中的跨站點腳本(XSS)?Apr 30, 2025 pm 03:38 PM

文章討論了通過輸入驗證,輸出編碼以及使用OWASP ESAPI和HTML淨化器之類的工具來防止PHP中的跨站點腳本(XSS)。

PHP中的自動加載是什麼?PHP中的自動加載是什麼?Apr 30, 2025 pm 03:37 PM

自動加載PHP會在需要時自動加載類文件,從而通過減少內存使用和增強代碼組織來提高性能。最佳實踐包括使用PSR-4和有效組織代碼。

什麼是PHP流?什麼是PHP流?Apr 30, 2025 pm 03:36 PM

PHP流通過一致的API來統一資源諸如文件,網絡插座和壓縮格式之類的處理,從而使復雜性抽象並增強代碼靈活性和效率。

可以使用PHP上傳的文件的最大大小是多少?可以使用PHP上傳的文件的最大大小是多少?Apr 30, 2025 pm 03:35 PM

本文討論了在PHP中管理文件上傳大小的管理,重點是2MB的默認限制以及如何通過修改PHP.INI設置來增加它。

PHP中的無效類型是什麼?PHP中的無效類型是什麼?Apr 30, 2025 pm 03:34 PM

本文討論了PHP 7.1中引入的PHP中的無效類型,允許變量或參數為指定類型或NULL。它突出顯示了諸如提高可讀性,類型安全性和明確意圖的好處,並解釋瞭如何聲明

unset()和unlink()函數之間有什麼區別?unset()和unlink()函數之間有什麼區別?Apr 30, 2025 pm 03:33 PM

本文討論了unset()和unlink()功能在編程中的差異,重點關注其目的和用例。 unset()從內存中刪除變量,而unlink()從文件系統中刪除文件。兩者都對效率至關重要

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

EditPlus 中文破解版

EditPlus 中文破解版

體積小,語法高亮,不支援程式碼提示功能

MinGW - Minimalist GNU for Windows

MinGW - Minimalist GNU for Windows

這個專案正在遷移到osdn.net/projects/mingw的過程中,你可以繼續在那裡關注我們。 MinGW:GNU編譯器集合(GCC)的本機Windows移植版本,可自由分發的導入函式庫和用於建置本機Windows應用程式的頭檔;包括對MSVC執行時間的擴展,以支援C99功能。 MinGW的所有軟體都可以在64位元Windows平台上運作。

DVWA

DVWA

Damn Vulnerable Web App (DVWA) 是一個PHP/MySQL的Web應用程序,非常容易受到攻擊。它的主要目標是成為安全專業人員在合法環境中測試自己的技能和工具的輔助工具,幫助Web開發人員更好地理解保護網路應用程式的過程,並幫助教師/學生在課堂環境中教授/學習Web應用程式安全性。 DVWA的目標是透過簡單直接的介面練習一些最常見的Web漏洞,難度各不相同。請注意,該軟體中

MantisBT

MantisBT

Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。