PHP类中的魔法方法有哪些

本文深入探讨PHP魔术方法,从构造器、析构器、属性操作到方法调用,逐一揭示其独特功能与应用。通过具体示例,展示了如何利用魔术方法简化对象生命周期管理与属性操作,实现自动化与高效编程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

看PHP.net
http://www.php.net/manual/zh/language.oop5.magic.php
这个标题有点牵强因为php有不只9种魔术方法,   但是这些将会引导你使用php魔术方法一个好的开始。它可能魔幻,但是并不需要魔杖。  
这些'魔术'方法拥有者特殊的名字,以两个下划线开始,表示这些方法在php特定事件下将会被触发。这可能听起来有点自动魔法但是它真的很酷的,我们已经看过一个简单的例子在  last post ,即我们使用一个构造器-使用这个作为我们第一个例子

__construct 

构造器是一个魔术方法,当对象被实例化时它会被调用。在一个类声明时它常常是第一件做的事但是没得必要他也像其他任何方法在类中任何地方都可以声明,构造器也能像其他方法样继承。如果我们想到以前继承例子从介绍到oop,我们能添加构造方法到Animal 类中,如:

class Animal{
 
  public function __construct() {
    $this->created = time();
    $this->logfile_handle = fopen('/tmp/log.txt', 'w');
  }
 
}
现在我们创建一个类来继承Animal类 - Penguin类!不添加任何属性和方法在Penguin类中,我们能申明并定义它继承自Animal类,如:  
class Penguin extends Animal {
 
}
 
$tux = new Penguin;
echo $tux->created;
如果我们定义一个构造方法在Penguin类中,然后Penguin对象将会运行当它被实例化后。由于并没有构造方法,PHP 会参考父类方法定义   信息来使用它因此我们能覆盖父类方法,或者不,在我们的新类中-很便利。

__destruct

你发现文件句柄也是构造器一部分吗?当我们使用完一个对象时真不想把事情放一边,因此析构方法做着与构造方法相反的事情。当对象被销毁时,析构方法会运行,或者明确的说当我们不再使用它时,php会为我们清理掉。Animal类中,我们的析构方法像这样,如:

class Animal{
 
  public function __construct() {
    $this->created = time();
    $this->logfile_handle = fopen('/tmp/log.txt', 'w');
  }
 
  public function __destruct() {
    fclose($this->logfile_handle);
  }
}
析构器让我们关闭任何额外的资源比如被使用过的对象。在php中由于我们有这样运行时间短的脚本(留意在更新的php版本中增强的垃圾回收机制),通常讨论内存溢出根本不需要。然而它仍是好的推行方法来清理而且总体上让程序运行起来更高效。 


__set

那么,我们将所有对 $this->name 的调用都更改为返回 $this->username的值,那么,如果我们想要设置这个值呢?也许我们有一个账户界面允许用户修改他们的名字。这时我们就需要 __set 方法的帮助了,举例说明:

class Penguin extends Animal {
 
  public function __construct($id) {
    $this->getPenguinFromDb($id);
  }
 
  public function getPenguinFromDb($id) {
    // elegant and robust database code goes here
  }
 
  public function __get($field) {
    if($field == 'name') {
      return $this->username;
    }
  }
 
  public function __set($field, $value) {
    if($field == 'name') {
      $this->username = $value;
    }
  }
}

这样,我们就针对大量的调用伪造对象的属性,正如我说的,这并不是一个正统的方法,但却是一个很有用的技巧,值得记住。



__get

这个魔术方法是一个非常灵巧的小技巧 - 它使实际上不存在的属性如同存在一半。让我们举个小企鹅的例子:

class Penguin extends Animal {
 
  public function __construct($id) {
    $this->getPenguinFromDb($id);
  }
 
  public function getPenguinFromDb($id) {
    // elegant and robust database code goes here
  }
}

现在,如果我们的小企鹅有一个 "name" 属性,而在此之后加载的属性为 "age",那么我们可以这样处理:

$tux = new Penguin(3);
echo $tux->name . " is " . $tux->age . " years old\n";

然而,设想一下,后端数据库或数据供应者发生了改变,"name"没有了,变味了"username"。并且设想这是一个非常复杂的应用,而需要修改的调用"name"的地方非常多。我们可以使用 __get 方法,使得"name"属性如同存在一样:

class Penguin extends Animal {
 
  public function __construct($id) {
    $this->getPenguinFromDb($id);
  }
 
  public function getPenguinFromDb($id) {
    // elegant and robust database code goes here
  }
 
  public function __get($field) {
    if($field == 'name') {
      return $this->username;
    }
}

这并不是编写整个系统的好方法,因为它会让调试工作变得更困难,但它是一个非常有价值的工具。它允许如同属性一样使用或者展示需要经过计算的数据,以及无数我都想不到的地方。


_call

这里有两种近似的方法,我并没有单独列出来,而是一起说明。一个是 _call 方法,如果定义,它将在调用未定义过的方法时被调用;另一个是 _callStatic 方法,工作方式与第一个相同,但却是在调用未定义的静态方法时生效(PHP 5.3 加入).通常我使用 __call 进行友善的错误处理,这在需要别人整合调用你的方法的库代码中非常有用。例如,如果一段脚本拥有一个企鹅对象,名为 $penguin ,它包含一个 $penguin->speak() 方法...假设 speak() 方法没有定义,那么正常情况下我们会看到:


PHP Fatal error: Call to undefined method Penguin::speak() in ...

通过定义 __call 方法,我们可以使用一些更友善的提示信息来代替 PHP 的错误提示:

class Animal {
}
class Penguin extends Animal {
 
  public function __construct($id) {
    $this->getPenguinFromDb($id);
  }
 
  public function getPenguinFromDb($id) {
    // elegant and robust database code goes here
  }
 
  public function __get($field) {
    if($field == 'name') {
      return $this->username;
    }
  }
 
  public function __set($field, $value) {
    if($field == 'name') {
      $this->username = $value;
    }
  }
 
  public function __call($method, $args) {
      echo "unknown method " . $method;
      return false;
  }
}

这将捕获的错误并回应。在实际应用中,更合适的方法是依据你的需要纪录消息日志·,将用户重定向,或者抛出一个异常,但概念是相同的。在这里你可以处理任何你需要处理的不当调用,你可以检测方法的名称,并一一处理——例如,你可以同上面我们重命名属性一些样重命名方法。



_sleep

__sleep()方法会被调用当对象被序列化后,并允许你处理序列化。这有各种各样的程序,一个很好的例子如果一个对象包含某种类型的指针,例如文件句柄或引用另一个对象。当对象被序列化然后解序列化,这些引用类型是无用的,因为这些类型的引用的目标可能不再存在或有效。因此,最好是来取消这些信息在存储它们之前。

__wakeup

__wakeup()是与__sleep()方法相反的,允许您更改对象解序列化的行为。和__sleep()一起使用,可以用来恢复被删除的句柄和对象当对象被序列化时。一个很好的例子程序是数据库句柄被取消设置当该项被序列化,然后恢复到当前配置中设置项目时,解序列化一个数据库句柄。


_clone

我们看过一个使用clone关键字的例子,在我的介绍从入门到oop的第二部分,创建对象的副本,而不是有两个变量指向同一个实际的数据。在一个类中重写此方法,我们可以观察发生了什么当在对象上使用clone关键字时,。虽然这是不是我们每一天能遇到的,一个漂亮的用例是创建一个真正的单例模式通过添加private访问修饰符给这个方法。

__toString

无疑把最好的始终留到最后,__toString方法是一个非常方便的附加方法对于我们的工具包。该方法可以声明覆盖对象的行为,当作为一个字符串输出时,例如,当它被输出时。如果你想能输出对象到模板中,你可以使用此方法来控制输出结果。让我们再来看看在Penguin类中:  
class Penguin {
 
  public function __construct($name) {
      $this->species = 'Penguin';
      $this->name = $name;
  }
 
  public function __toString() {
      return $this->name . " (" . $this->species . ")\n";
  }
}
在适当的位置,输出该对象通过调用echo输出它,如:  
$tux = new Penguin('tux');
echo $tux;
我不常常使用这种捷径,但是知道它的存在是很有用的。  

常用的魔术方法有:__Tostring ()  __Call()  __autoLoad()  __ clone()  __GET()   __SET()    __isset()  __unset()
 
1.__Tostring()   用于定义输出对象引用时调用  常用于打印一些对象的信息 必须有返回值
eg:有一个persion类
Persion per =new persion()
Echo per;    //直接调用会出错
我们可以在类的定义中添加__tostring()方法

复制代码代码如下:

Function  __Tostring()
{
$str=this->$name.this->age;
Return $str;
}

2.__clone()对象的复制
引用赋值
$per1=$per2; 而这在内存中只有一块地址
而$per1=clone $per2   这时有两块内存地址

3.__call()方法 当调用类实例中不存在的函数时自动执行
如果试图调用类中不存在的函数,会出现语法错误,为了能够友好的提示
我们可以在类中声明Call()方法;
复制代码代码如下:

Function __call($funName,$argu)
{
Echo "名为".$funName."参数为".printf($argh)."的函数不存在",
}

4.__autoLoad 自动加载使用的类文件  该函数是在引用的页面添加
我们都使用过这样情况,在页面中需要调用其他php文件,我们需要使用include方法
但是如果有几十个页面需要引用,未免太过繁琐,我们可以在该页面中使用autoload方法
复制代码代码如下:

Function __autoload($className)
{
Include $className.".php";
}

这样凡是引用到其他类的地方,都会自动引用该类文件  前提类文件的名称必须是   类名.php

5.__GET()   访问类中私有属性
如果类中的属性设置为私有属性,在类的实例中是无法访问的,但怎样才能访问呢?
我们就可以使用__GET()
Eg :
类中有

复制代码代码如下:

Class person
{
Private $name;
Private $age;
}

实例化 person per=new person()
Per->$name; //这样是取不到值的
但是如果我们在类中增加__GET方法
复制代码代码如下:

Function __GET($proName)
{
Return this->$proName;
}

我们再次调用Per->$name 就可以访问了
这样做有人会提出疑问了,这样可以直接访问私有变量,和声明成公有的有什么区别呢?
如果声明成公有的,我们可以任意读取,如果是私有,如果我们增加了get方法,那么每次调用私有属性都会调用GET方法的内容,这样我们就可以在get方法中增加一些逻辑处理。

6.__SET()设置类中的私有属性
原理同上,我们可以再类中添加__SET()函数,每当通过调用类实例给私有属性赋值,都会执行__SET函数 ,函数原型:

复制代码代码如下:

Function __SET($proName,$value)
{
This->$proName=$value;
}

既然是方法赋值,我们就可以做一些逻辑处理

7.__isset() 判断类中私有属性或方法是否存在时自动调用
首先我们先介绍一下isset 方法,该方法用于判定属性和方法是否存在,但是我们无法通过类类实例判断类中的某个私有属性是否存在 
如果我们使用isset(per->$name);//返回值是false,但是$name属性的确存在,怎么解决呢?
解决方法:
1.将$name定义为私有属性
2.在类定义中添加

复制代码代码如下:

Function __isset($proName)
{
Return  isset(this->$proName);//再类内部是可以访问私有属性的
}

这样的话我们再次调用isset($name);返回值就为true了;

8.__unset()清除类中私有变量时自动调用
与之结合的是unset() unset方法可以删除属性,当我们需要删除类中属性的时候,如果是公有属性我们可以直接
删除,但是如果是私有我们只通过该方法就无法实现了
怎样实现呢我们可以使用__unset()方法实现该功能我们需要在类中添加

复制代码代码如下:

Function __unset($proName)
{
Unset(this->$proName);
}

现在我们再调用unset($name);就可以删除person类中的私有属性$name了
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值