上一篇介绍了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(),让我们可以对于一个“不存在的”属性,来进行删除操作。
总体来说,啰啰嗦嗦的这一节讲述了不存在的方法和属性,大家可以想象一下我们在使用一些框架的时候,是不是经常会对一些不存在的属性(方法)进行调用呢? 好了今天写到这里,欢迎批评指正,或者留言一起探讨。