Maison >développement back-end >tutoriel php >php类中的魔术方法及类的自动加载

php类中的魔术方法及类的自动加载

WBOY
WBOYoriginal
2016-07-25 09:03:52980parcourir
  1. class Connection {
  2. protected $link;
  3. private $server, $username, $password, $db;
  4. public function __construct($server, $username, $password, $db) {
  5. $this->server = $server;
  6. $this->username = $username;
  7. $this->password = $password;
  8. $this->db = $db;
  9. $this->connect();
  10. }
  11. private function connect() {
  12. $this->link = mysql_connect($this->server, $this->username, $this->password);
  13. mysql_select_db($this->db, $this->link);
  14. }
  15. public function __sleep() {//系列化时只需保存这些属性的值
  16. return array('server', 'username', 'password', 'db');//必须返回一个数组
  17. }
  18. public function __wakeup() { //反系列化得到对象后立即连接数据库
  19. $this->connect();
  20. }
  21. }
复制代码

__toString(): 试图将对象作为一个字符串使用时被调用,返回一个字符串,类似 js 对象的 toString()

__invoke(): 将一个对象作为函数调用时被调用。如 $object($a,$b) 将调用 $object->__invoke($a,$b)

__set_state() : 对一个对象使用 var_export() 时被调用,其返回值将被打印

__clone(): 对一个对象使用 clone 操作时被调用,无返回值,该方法可用于 在返回 clone 得到的对象之前修改其属性使得 clone 得到的对象与 被操 作的对象属性值可以不同,但并不能通过返回 null 或 false 来阻止 对象克隆操作, clone 操作得到的对象不是通过该方法的返回值返回 的。

注意事项: __get、__set 方法可常用于灵活处理 对象私有属性、保护属性的访问。 由于 PHP 对象对于的方法使用 isset() 或 empty() 判定时,不会认为 方法是一个可访问的属性(写多了 javascript 要注意了,PHP 里面 属性就是属性方法就是方法不可混淆) 在使用 __get 时 ,你可能会在该方法中对属性使用 isset($this->key) 判断,尤其是要特别处理 私有属性的时候,这时要注意 __set() 有没有定义以及如何定义以免出现误判。 如果一个对象的方法在调用时不需要传参,或者传参是固定的,可以通过 __get() 方法来将方法属性化,在 __get() 里面自动调用该方法并将值返回。

比如在类内定义 __get 方法如下:

  1. public function __get($key){

  2. if(property_exists($this,$key)){
  3. return $this->$key; //私有、保护属性允许访问
  4. }else if(method_exists($this,'get'.$key)){
  5. return $this->{'get'.$key}();
  6. //或者....
  7. // $methodName = 'get'.$key;
  8. //return $this-> $methodName();
  9. }else{
  10. thrown new Exception('class '.__class__.' do not has property '.$key);
  11. }
  12. }
  13. //方法属性化访问:

  14. $obj->getModelName();
  15. $obj->ModelName; //属性化
复制代码

unserialize() 方法:当试图反序列化一个对象时,该函数需要知道对象的 类,如果这个系列化字符串从其他方式得来,脚本环境中没有定义对象的类,就需要将类的文件引入,unserialize() 第二个参数是可选的 callback 型参数,用于引入类所在的文件。 function importClass($calssName){ include('xxxx.php'); //包含该类的文件 } unserialize($objstr,$callbackName);

类的自动加载: __autoload()

__autoload() 是PHP执行环境中约定的一个函数而非某个类的方法,如果一个类在使用之前没有加载到当前文件,会自动调用 __autoload() 函数来加载该类,通常这些类的加载规则都是约定的,比如这些类包含在以类名命名的文件内,该方法可以实现类的按需加载,避免脚本执行前加载不必要的类从而降低资源占用、提交性能。

注意:__autoload() 内的错误不能被 try-catch 捕获。

  1. function __autoload($class_name){
  2. require_once(PATH.'/calsses/'.$class_name.'.php');
  3. }
  4. $obj1 = new mycalss1();
复制代码

注册 __autoload() 自动调用的函数: spl 代码库在 PHP5.0 之后默认自动启用 spl_autoload_register([callback]); //不将具体实现的加载代码写在 __autoload() 内,可使用该函数注册回调函数。

如果使用类的方法作为回调函数需要传入一个数组: spl_autoload_register(array('class_name'|$obj,'method_name'));

例如: spl_autoload_register(array($this,'autoloadClass')); spl_autoload_register(array('YiiBase','autoload'));// YII 框架的自动加载类的实现, YiiBase 类实现了一个autoload 方法。 spl_autoload_register() 可以注册多个加载函数,成功加载类文件之前将逐个尝试所有注册的加载函数。这在不同的类使用不同逻辑来导入类文件的时候很有用。 spl_autoload_unregister(); //取消某个注册的加载函数,参数与 spl_autoload_register() 相同. spl_autoload_functions();// 以数组形式返回所有注册的 __autoload() 函数

spl_autoload(class_name[,file_extentions]); // __autoload() 函数的默认实现。 spl_autoload_register() 被调用时如果没有传入 函数名,则默认使用该函数,该函数的执行规则是: 类名转为小写作为文件名,传入的 file_extentions(多个扩展名以逗号隔开,默认为 .inc 和 .php)为扩展名,根据得到的文件名尝试在 php.ini 内设置的 include paths 中搜索。

spl_autoload_call(class_name);//手动调用所有注册的 __autoload() 函数来主动加载类文件 spl_autoload_extentions([file_extentions]); //注册或返回 spl_autoload() 中可以使用的文件扩展名,扩展名可以是 .a.b 这样的形式,例如: spl_autoload_extentions(".class.php"); spl_autoload_register(); //使用spl_autoload() 来尝试自动加载类文件 //这样 spl_autoload('myclassName'); 会尝试加载 文件 "myclassName.class.php" .



Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn