欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

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

程序员文章站 2022-05-15 21:19:18
...
  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" .