上一篇介绍了PHP中的魔术常量和魔术方法__construct(),下面继续介绍PHP中的魔术方法。(大神继续跳过) __destruct(),类的析构函数,跟__construct()相似。析构方法让我们可以再销毁类之前执行一些操作或完成一些功能,比如释放资源、结果集,关闭文件等等。这个就不做解释了
下面的几个比较重要,在后边的设计模式中,会有很多应用,熟练掌握这些方法,对于提高编程能力有很大帮助。 __call(),当我们调用一个不可访问的方法时调用。这个方法接收两个参数,第一个参数 functionname会自动接收不存在的方法名,第二个参数 argument则以数组的方式接收不存在方法的多个参数。 __callStatic,跟call()相似,在静态上下文中调用一个不可访问的方法时,这个方法就会被调用。 让我们看一个官方的例子
class MethodTest { public function __call($name, $arguments) { // 注意: $name 的值区分大小写 echo "Calling object method '$name' " . implode(', ', $arguments). "\n"; } /** PHP 5.3.0之后版本 */ public static function __callStatic($name, $arguments) { // 注意: $name 的值区分大小写 echo "Calling static method '$name' " . implode(', ', $arguments). "\n"; } } $obj = new MethodTest; $obj->runTest('in object context'); MethodTest::runTest('in static context'); // PHP 5.3.0之代码会输出: Calling object method ‘runTest’ in object context Calling object method ‘runTest’ in static context
get(), 这里官方的说法是,读取不可访问的属性值时,__get()会调用,对比一下set() set(),在给不可访问的属性赋值时,__set()会被调用。 那么什么是不可访问?PHP中,如果我们把一个属性值设置为私有的,那么如果我们在外部访问,就会报错,或者当这个属性根本就没有在类中设置的时候,对于PHP来说,也就是不可访问的属性了。看下面的例子:
class PropertyTest { //我们将在这里保存被重载的数据,也就是不可访问的属性 private $data = []; /** 重载不能被用在已经定义的属性 */ public $declared = 1; /** 只有从类外部访问这个属性时,重载才会发生 */ private $hiiden = 2; public function __set($name, $value) { echo 'setting '. $name. ' to '. $value; $this->data[$name] = $value; } public function __get($name) { echo 'getting '. $name; if (array_key_exists($name, $this->data)){ return $this->data[$name]; } $trace = debug_backtrace(); trigger_error( ‘Undefined property via __get() ’. $name. ' in '. $trace[0]['file']. ' on line '. $trace[0]['line'], E_USER_NOTICE ); return null; } /**php5.1之后**/ public function __isset($name) { echo 'Is '. $name. ' set?'; return isset($this->data[$name]); } public function __unset($name) { echo 'Unsetting '. $name; unset($this->data[$name]); } /** 非魔术方法 **/ public function getHidden() { return $this->hidden; } $obj = new PropertyTest; $obj->a = 1; echo $obj->a . "\n\n"; var_dump(isset($obj->a)); unset($obj->a); var_dump(isset($obj->a)); echo "\n"; echo $obj->declared . "\n\n"; echo "Let's experiment with the private property named 'hidden':\n"; echo "Privates are visible inside the class, so __get() not used...\n"; echo $obj->getHidden() . "\n"; echo "Privates not visible outside of class, so __get() is used...\n"; echo $obj->hidden . "\n"; } 上面的代码会输出 Setting 'a' to '1' Getting 'a' 1 Is 'a' set? bool(true) Unsetting 'a' Is 'a' set? bool(false) 1 Let's experiment with the private property named 'hidden': Privates are visible inside the class, so __get() not used... 2 Privates not visible outside of class, so __get() is used... Getting 'hidden' Notice: Undefined property via __get(): hidden in <file> on line 70 in <file> on line 29可以看到,当我们给obj对象不存在的属性‘a’赋值的时候,这个值被保存到data这个数组中了,而当我们想要获取的时候,直接通过obj->a就从data数组中把a的值取出来了。值得注意的是,在5.1版本之后,当我们想知道一个属性(注意也是不存在的,我们可以用这个方法来验证我们之前是否定义过)是否存在的时候,调用isset这个函数的时候,类中的__isset方法会被调用,相对应的还有usset(),让我们可以对于一个“不存在的”属性,来进行删除操作。 总体来说,啰啰嗦嗦的这一节讲述了不存在的方法和属性,大家可以想象一下我们在使用一些框架的时候,是不是经常会对一些不存在的属性(方法)进行调用呢? 好了今天写到这里,欢迎批评指正,或者留言一起探讨。