abei1982 2017-05-23 00:10:21 7108次浏览 11条回复 12 1 0

书接上回,我将代码提交到服务器后被小X经理一顿批,她是这样说的。

小XX: 你丫不知道yii2自己就有登陆后的事件么陆后的事件么的事件么件么么我不知道啊道啊啊咋的吧的吧吧小XX:改!然后统计个yii2预定义事件表给我i2预定义事件表给我事件表给我表给我给我我我哦好的

好吧,算咱技术不到家,那就赶紧改吧,先回顾下上一篇我都干了啥,我是在会员登陆后触发了一个事件,既然说有内置的,那就找找吧。

我知道登陆功能是使用yii\web\User这个类,那应该去那里去找它们。它在@app/vendor/yiisoft/yii2/web/User.php

@@nai8@@

5分钟以后~

果然,我找到了那个事件,还用了3分钟分析了如何使用,现在把我的研究成果和大家分享下。

对于yii2系统的登陆,yii\web\User类一共提供了4个事件,如下

const EVENT_BEFORE_LOGIN = 'beforeLogin';
const EVENT_AFTER_LOGIN = 'afterLogin';
const EVENT_BEFORE_LOGOUT = 'beforeLogout';
const EVENT_AFTER_LOGOUT = 'afterLogout';

中彩网分别是登陆前后、注销前后,我要使用的是EVENT_AFTER_LOGIN事件,那么如何去使用那?

既然是登陆,还是先看下yii\web\User中的login方法,看看是否有蛛丝马迹

public function login(IdentityInterface $identity, $duration = 0){

    if ($this->beforeLogin($identity, false, $duration)) {
        $this->switchIdentity($identity, $duration);
        $id = $identity->getId();
        $ip = Yii::$app->getRequest()->getUserIP();
        if ($this->enableSession) {
            $log = "User '$id' logged in from $ip with duration $duration.";
        } else {
            $log = "User '$id' logged in from $ip. Session not enabled.";
        }
        Yii::info($log, __METHOD__);
        $this->afterLogin($identity, false, $duration);
    }

    return !$this->getIsGuest();
}    

我看到了 $this->afterLogin($identity, false, $duration),似乎这个方法是登陆后做了一些事情,那就继续看吧

protected function afterLogin($identity, $cookieBased, $duration){

    $this->trigger(self::EVENT_AFTER_LOGIN, new UserEvent([
        'identity' => $identity,
        'cookieBased' => $cookieBased,
        'duration' => $duration,
    ]));
}    

中彩网OMG,真的发现了,原来这个方法触发了EVENT_AFTER_LOGIN事件,并且还很贴心的将identity等信息传递给了绑定EVENT_AFTER_LOGIN事件的每一位观察者。

于是思路来了

我只需要在login前绑定EVENT_AFTER_LOGIN事件,然后afterLogin会自动触发所有订阅者。

重构开始

use yii\web\User;
class UserController extends Controller {
	
	public function __construct(){
		//	绑定事件
		$this->on(User::EVENT_AFTER_LOGIN,['app\models\OLog','add']); 
		$this->on(User::EVENT_AFTER_LOGIN,['app\models\Admin','sendMail']); 
		$this->on(User::EVENT_AFTER_LOGIN,['app\models\User','notifyFirend']); 
	}
	
	public function actionIndex(){
		.....
		//	login				
	}
}

我之前定义的EVENT_USER_LOGIN自然可以删除了,上面看到afterLogin方法触发事件的时候已经传递了一个UserEvent,里面含有登陆的详细信息了,我上一篇自己定义的那个UserLoginEvent也就可以删除了。

中彩网使用系统自带的事件,真心省了太多代码了,感谢yii2开发团队,感谢你们八倍祖主。

中彩网对于登陆主题就重构完了,现在我还需要改下订阅者的代码,毕竟传递给订阅者的$event不一样了,拿一个订阅者举例吧

// User app\models\User.php
class User {
    static public function notifyFirend($event){
        $userId = $event->userId;
        echo "告诉了朋友们我登陆了";
    }
}

我要对其进行小手术,diu一下,手术后的样子如下

// User app\models\User.php
class User {
    static public function notifyFirend($event){
        $userId = $event->identity->id;
        echo "告诉了朋友们我登陆了";
    }
}

ok~ 到此为止我们对登陆逻辑的事件处理就完成了,去掉了自定义的事件,绑定了 User::EVENT_AFTER_LOGIN 内置事件。

还没完~

是的还没完,小X经理还让我提交一个yii2预定义事件表给她,我深刻体会到这个表格的重要性,有了它,便可以通晓yii所有事件,正所谓就算成功,也不一定自宫。

但是,yii2这么大,我如何一个不落的得到它们那?

看来要借助于我们的神器PHPSTORM了。

首先我们知道yii2对于事件名的命名规则,常量大写、观察下发现内置的这些事件都是EVENT_开头的中彩网,有了这些特征,那开始吧。

方法如下:中彩网我们对vendor/yiisoft目录进行查找(find in path),找到所有const EVENT_开头的行。

Snip20170522_34.png

表之event内置一览表

建议你背下这些事件,至少在做事件的时候查一遍速查表,能用系统的优先使用系统的。

学到了一些

很高兴完成了小x经理关于登陆事件的需求,通过这些让我对yii的事件有了一个大体了解,知道了事件是观察者模式的一种实现、知道yii2内置了很多事件,我们应该优先使用它们。

begin git。

中彩网一阵QQ震动,带着坏笑的小x经理发了一句消息给我:“来,小北同学!”

天,又咋了,下篇告诉你。

觉得很赞
  • 回复于 2017-05-24 12:22 举报

    按你说的报错啊,

    $this->on(User::EVENT_AFTER_LOGIN,['app\models\OLog','add']); 
    $this->on(User::EVENT_AFTER_LOGIN,['app\models\Admin','sendMail']); 
    $this->on(User::EVENT_AFTER_LOGIN,['app\models\User','notifyFirend']); 
    

    可以这样吗

    1 条回复
    回复于 2017-05-24 12:26 回复

    use yii\web\User; 了么,什么报错信息

  • 回复于 2017-05-24 15:45 举报

    中彩网不报错了,但是怎么知道这个方法执行了呢,

    static public function notifyFirend($event){
        $userId = $event->userId;
        echo "告诉了朋友们我登陆了";
    }
    

    我也自定义了这个方法,就是不执行啊

    3 条回复
    回复于 2017-05-24 22:25 回复

    中彩网运行触发事件的控制器,就会执行

    回复于 2017-05-24 22:27 回复

    在方法中 $this->trigger(),这样触发时间

    回复于 2017-05-24 22:30 回复

    要先绑定,建议先看我上一篇文章,先搞懂观察者模式,然后在看事件机制,搞懂原理。
    我的yii视频小站 nai8.me

  • 回复于 2017-05-24 22:29 举报

    对了,你怎么晓得他是观察者模式呢 ????

    1 条回复
    回复于 2017-05-24 22:31 回复

    给强哥发了一个facebook留言

  • 回复于 2017-08-09 10:46 举报
    $this->trigger(self::EVENT_AFTER_LOGIN, new UserEvent([
        'identity' => $identity,
        'cookieBased' => $cookieBased,
        'duration' => $duration,
    ]));
    

    请问你这个登录后触发 EVENT_AFTER_LOGIN 事件, new UserEvent 这个又是什么? 你文章没有这个类啊
    绑定的事件触发 User 下的 notifyFirend 的方法 这个 new UserEvent 是带入到 notifyFirend($event)的实例化参数吗

    class User {
        static public function notifyFirend($event){
            $userId = $event->identity->id;
            echo "告诉了朋友们我登陆了";
        }
    }
    
  • 回复于 2017-11-13 16:48 举报

    中彩网请教:使用basic模板时,User.php是继承了\yii\base\Object,这些没法弄?

    觉得很赞
  • 回复于 2017-11-16 09:15 举报
    • 请教: afterLogin触发时,事件的订阅者(绑定的事件处理handle)并不会被通知到,此时,绑定了handle的_events数组在identity里面的_events中,$this已经变化了,$this中的events(没有包含我们绑定的handle)才会被触发到,请问如何解决呢?
    • 如果用自定义事件来触发
    • $this->trigger(self::EVENT_USER_LOGIN);
    • $this触发时和绑定时的$this是一样的,此时绑定的handle正好在$this的_events数组中,当然可以触发的,没有问题。
    • 如果用内置的事件来触发
    • Yii::$app->user->login($user);
    • 触发afterLogin时,$this已经变成了yii\web\Use
    • 999.png
  • 回复于 2017-11-16 10:09 举报

    搞定了~~

    觉得很赞
  • 回复于 2017-12-12 15:42 举报

    中彩网哥们,你有写故事的潜质啊!哈哈,看着挺乐呵的。

  • 回复于 2017-12-26 18:12 举报
  • 回复于 2019-11-29 15:14 举报

    在controller的Action中写的绑定事件,之后事件不会执行,

    请问题主的controller继承的是哪个呢?我继承的是yii\web\Controller。
    /**

     * @param string $id the ID of this controller.
     * @param Module $module the module that this controller belongs to.
     * @param array $config name-value pairs that will be used to initialize the object properties.
     */
    public function __construct($id, $module, $config = [])
    {
        $this->id = $id;
        $this->module = $module;
        parent::__construct($config);
    }
    
    2 条回复
    回复于 2019-11-29 15:16 回复

    中彩网难道必须要把on写到构造函数里?没道理啊

    回复于 21小时前 回复

    中彩网 根据我自己实测的经验,使用$this->on绑定并不会触发绑定事件,使用yii::$app->user->on的绑定可以正常触发登录事件

  • 回复于 21小时前 举报

    根据我自己实测的经验,使用$this->on绑定并不会触发绑定事件,使用yii::$app->user->on的绑定可以正常触发登录事件

您需要登录后才可以回复。登录 | 立即注册