php中的魔术方法
2016-08-21 00:32
127 查看
魔术方法:
是指某些情况下,会自动调用的方法,称为魔术方法
PHP面向对象中,提供了这几个魔术方法,
他们的特点 都是以双下划线__开头的
__construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(), __sleep(), __wakeup(), __toString(), __invoke(), __set_state() 和 __clone()
__construct :构造方法
__destruct :析构方法
__clone() :克隆方法,当对象被克隆时,将会自动调用
接下来我们利用魔术方法来做改进:
当我们调用一个权限上不允许调用的属性,和不存在的属性时,
__get魔术方法会自动调用,
并且自动传参,参数值是属性名.
流程:
生活中,你帮别人看守小卖店
买牙刷—>好,给你牙刷
买毛巾—>好,给你毛巾
这个POS机挺好—> (pos是商店的工具,私有的,不卖的:”你无权买”), 但是我们用__get方法,
就有一个友好的处理机会.
系统会直接报错,甚至fatal error,通过__get,我们就能自定义用户访问时,的处理行为.
改进:
按我们以前的做法,
把POST来的数据,拼接sql,然后查询.
但是在TP中的做法,有点奇怪,
他是把收到的信息,
赋给了一个对象的属性.
然后对象->add()方法,就写入到数据库了.
很方便 .
思考:
1: userModel就有username属性供你去赋值吗?
2: 如果userModel−>xxx属性,是保护的,而我的表,又有一个字段,恰好也叫xxx,那么我自然是user->xxx = $_POST[‘xxx’];
这不就出错了吗?
3:还有一个问题: userModel 有一些属性,很正常,比如有5个属性
a,b,c,d,e
我在注册时, 又动态设置了属性, f,g,h,i
疑问: 在拼接sql时,要把a,b,c,d,e忽略掉才行.
又怎么忽略.
答:用魔术方法来解决
通过__set()方法,
把属性的设置—>都放到数组里.
处理时,专门处理这个数组就可以了.
这样,就不会和其他属性相冲突
*/
/*
TP中的一段用户注册代码
*/
是指某些情况下,会自动调用的方法,称为魔术方法
PHP面向对象中,提供了这几个魔术方法,
他们的特点 都是以双下划线__开头的
__construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(), __sleep(), __wakeup(), __toString(), __invoke(), __set_state() 和 __clone()
__construct :构造方法
__destruct :析构方法
__clone() :克隆方法,当对象被克隆时,将会自动调用
__clone()
class Human{ public age=22; public function __clone(){ echo "有人要克隆我,是假的!"; } } $lisi=new Human(); $zhangsan=clone $lisi; //"有人要克隆我,是假的!"
接下来讲6个,在项目中,尤其是自己想写框架时,很实用的几个函数
__call(), __callStatic(), __get(), __set(), __isset(), __unset(),01 __get()
class Human{ private $money="30两"; protected $age="24岁"; public $name="lisi"; } $lisi=new Human(); echo $lisi->name; //lisi 正常输出 echo $lisi->age; //权限错误 echo $lisi->friend; //出错,未定义
接下来我们利用魔术方法来做改进:
class Human{ private $money="30两"; protected $age="20"; public $name="lisi"; public function __get($p){ echo "你想访问我的".$p."属性"; } } $lily = new Human(); // echo $lily->name; //lily echo $lily->age; // '你想访问我的age属性 :) echo $lily->money; // 你想访问我的money属性 :) echo $lily->friend; // 你想访问我的friend属性 :)
总结:
可以总结出:当我们调用一个权限上不允许调用的属性,和不存在的属性时,
__get魔术方法会自动调用,
并且自动传参,参数值是属性名.
流程:
$lily->age--无权-->__get(age); $lily->friend--没有此属性-->__get('friend');
生活中,你帮别人看守小卖店
买牙刷—>好,给你牙刷
买毛巾—>好,给你毛巾
这个POS机挺好—> (pos是商店的工具,私有的,不卖的:”你无权买”), 但是我们用__get方法,
就有一个友好的处理机会.
系统会直接报错,甚至fatal error,通过__get,我们就能自定义用户访问时,的处理行为.
02 __set()
class Human{ private $money="30两"; protected $age="24岁"; public $name="lisi"; } $lily->aaa = 111; $lily->bbb = 222; print_r($lily);//aaa,bbb两个属性竟然都给加上了.其实,对象就是一个属性集,在js中更明显.但是如果这么随便就能加了属性,导致这个对象属性过多,不好管理
改进:
class Stu { private $money = '30两'; protected $age = 23; public $name = 'Hmm'; public function __set($a,$b) { echo '你想设置我的',$a,'属性','<br />'; echo '且值是',$b,'<br />'; } } $hmm = new Stu(); $hmm->aaa = 111; $hmm->money = '40两'; $hmm->age = '28'; print_r($hmm); $hmm->name = 'HanMM'; print_r($hmm); /* 如上,总结出 __set的作用 当为无权操作的属性赋值时, 或不存在的属性赋值时, __set()自动调用 且自动传2个参数 属性 属性值 例: $hmm->age = 28 ---无权---> __set('age',28); */
03 __isset() __unset()
class Dog { public $leg = 4; protected $bone = '猪腿骨'; public function __isset($p) { echo '你想判断我的',$p,'属性存不存在<br />'; return 1; } public function __unset($p) { echo '你想去掉我的',$p,'?!<br />'; } } $hua = new Dog(); if(isset($hua->leg)) { echo $hua->leg; } if(isset($hua->tail)) { echo '有tail属性'; } else { echo '没有tail'; } /*** __isset() 方法, 当 用isset() 判断对象不可见的属性时(protected/private/不存在的属性) 会引发 __isset()来执行 问: isset($obj->xyz) 属性,为真, 能说明 类声明了一个xyz属性吗? 答:不能 ***/ echo '<hr />'; echo '__unset测试'; print_r($hua); unset($hua->leg); print_r($hua); unset($hua->tail); unset($hua->bone); /*** __unset()方法 当 用unset 销毁对象的不可见属性时, 会引发 __unset(); unset($hua->tail)----没有tail属性---->__unset('tail');
魔术方法在thinkphp中的使用
刚才 我们用TP做了一个用户注册,注册时的代码在下面按我们以前的做法,
把POST来的数据,拼接sql,然后查询.
但是在TP中的做法,有点奇怪,
他是把收到的信息,
赋给了一个对象的属性.
然后对象->add()方法,就写入到数据库了.
很方便 .
思考:
1: userModel就有username属性供你去赋值吗?
2: 如果userModel−>xxx属性,是保护的,而我的表,又有一个字段,恰好也叫xxx,那么我自然是user->xxx = $_POST[‘xxx’];
这不就出错了吗?
3:还有一个问题: userModel 有一些属性,很正常,比如有5个属性
a,b,c,d,e
我在注册时, 又动态设置了属性, f,g,h,i
疑问: 在拼接sql时,要把a,b,c,d,e忽略掉才行.
又怎么忽略.
答:用魔术方法来解决
通过__set()方法,
把属性的设置—>都放到数组里.
处理时,专门处理这个数组就可以了.
这样,就不会和其他属性相冲突
*/
/*
TP中的一段用户注册代码
$userModel->username = $_POST['username']; $userModel->email = $_POST['email']; if($num = $userModel->table('user')->add()) { echo '注册成功'; } else { echo 'fail'; }
*/
class UserModel { protected $email = 'user@163.com'; protected $data = array(); public function __set($k,$v) { // $this->$k = $v; //并没有真正赋成自己的属性 $this->data[$k] = $v; // 而是放在一个数组里 } public function __get($p) { return isset($this->data[$p]) ? $this->data[$p] : NULL; } public function __unset($p) { unset($this->data[$p]); } public function __isset($p) { return isset($this->data[$p]); } public function add() { $sql = 'insert into table ('; $sql .= implode(',',array_keys($this->data)); $sql .= ') values (\''; $sql .= implode("','",array_values($this->data)); $sql .="')"; return $sql; } } echo '<pre>'; $userModel = new UserModel(); print_r($userModel); $userModel->username = 'lisi'; $userModel->email = 'lisi@126.com'; print_r($userModel); //echo $userModel->add(); unset($userModel->email); print_r($userModel);
相关文章推荐
- 讲讲php的魔术方法
- php魔术方法
- PHP 5魔术方法应用浅析
- php 魔术方法使用说明详细
- php 魔术方法
- php的魔术方法
- PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep, __wakeup, __toStr
- PHP下划线函数即魔术方法汇总
- PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __uns
- PHP中用魔术方法__autoload()实现类的自动加载
- 有了PHP 5.3的__callStatic魔术方法,就可以做按类名和方法名同时魔术了
- php魔术方法
- php魔术方法和魔术变量
- 腾讯php面试题总结(一) 15个魔术方法的总结
- PHP中的魔术方法总结
- php 魔术方法使用说明
- PHP中的魔术方法总结 :__construct,__destruct ,__call,__callStatic,__get,__set,__isset,__unset,__sleep,__wakeup, __toString.......
- PHP中的魔术方法总结[__construct/__destruct/__get/__set/__isset/__unset/__toString/__clone等]
- php魔术方法和魔术变量、内置方法和内置变量
- PHP魔术方法