php中的魔术方法

魔术方法:
是指某些情况下,会自动调用的方法,称为魔术方法
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);
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值