目录
Yii2的深入学习--yii\base\Event 类,yii2--yii
首页 后端开发 php教程 Yii2的深入学习--yiibaseEvent 类,yii2--yii_PHP教程

Yii2的深入学习--yiibaseEvent 类,yii2--yii_PHP教程

Jul 12, 2016 am 09:02 AM
yii

Yii2的深入学习--yii\base\Event 类,yii2--yii

根据之前一篇文章,我们知道 Yii2 的事件分两类,一是类级别的事件,二是实例级别的事件。类级别的事件是基于 yii\base\Event 实现,实例级别的事件是基于 yii\base\Component 实现。

今天先来看下类级别事件的实现,代码是 yii\base\Event 类。

<?<span>php
namespace yii\base;

</span><span>/*</span><span>*
 * Event is the base class for all event classes.
 </span><span>*/</span>
<span>class</span> Event <span>extends</span> <span>Object</span><span>
{
    </span><span>/*</span><span>*
     * @var string the event name. This property is set by [[Component::trigger()]] and [[trigger()]].
     * Event handlers may use this property to check what event it is handling.
     * 事件的名字
     </span><span>*/</span>
    <span>public</span> <span>$name</span><span>;
    </span><span>/*</span><span>*
     * @var object the sender of this event. If not set, this property will be
     * set as the object whose "trigger()" method is called.
     * This property may also be a `null` when this event is a
     * class-level event which is triggered in a static context.
     * 触发事件的对象
     </span><span>*/</span>
    <span>public</span> <span>$sender</span><span>;
    </span><span>/*</span><span>*
     * @var boolean whether the event is handled. Defaults to false.
     * When a handler sets this to be true, the event processing will stop and
     * ignore the rest of the uninvoked event handlers.
     * 记录事件是否已被处理,当 handled 被设置为 true 时,执行到这个 event 的时候,会停止,并忽略剩下的 event
     </span><span>*/</span>
    <span>public</span> <span>$handled</span> = <span>false</span><span>;
    </span><span>/*</span><span>*
     * @var mixed the data that is passed to [[Component::on()]] when attaching an event handler.
     * Note that this varies according to which event handler is currently executing.
     </span><span>*/</span>
    <span>public</span> <span>$data</span><span>;

    </span><span>/*</span><span>*
     * 存储所有的 event,因为是 static 的属性,所有的 event 对象/类都共享这一份数据
     </span><span>*/</span>
    <span>private</span> <span>static</span> <span>$_events</span> =<span> [];


    </span><span>/*</span><span>*
     * Attaches an event handler to a class-level event.
     *
     * When a class-level event is triggered, event handlers attached
     * to that class and all parent classes will be invoked.
     *
     * For example, the following code attaches an event handler to `ActiveRecord`'s
     * `afterInsert` event:
     *
     * ~~~
     * Event::on(ActiveRecord::className(), ActiveRecord::EVENT_AFTER_INSERT, function ($event) {
     *     Yii::trace(get_class($event->sender) . ' is inserted.');
     * });
     * ~~~
     *
     * The handler will be invoked for EVERY successful ActiveRecord insertion.
     *
     * For more details about how to declare an event handler, please refer to [[Component::on()]].
     *
     * 为一个类添加事件
     *
     * @param string $class the fully qualified class name to which the event handler needs to attach.
     * @param string $name the event name.
     * @param callable $handler the event handler.
     * @param mixed $data the data to be passed to the event handler when the event is triggered.
     * When the event handler is invoked, this data can be accessed via [[Event::data]].
     * @param boolean $append whether to append new event handler to the end of the existing
     * handler list. If false, the new handler will be inserted at the beginning of the existing
     * handler list.
     * @see off()
     </span><span>*/</span>
    <span>public</span> <span>static</span> <span>function</span> on(<span>$class</span>, <span>$name</span>, <span>$handler</span>, <span>$data</span> = <span>null</span>, <span>$append</span> = <span>true</span><span>)
    {
        </span><span>//</span><span> 去掉 class 最左边的斜杠</span>
        <span>$class</span> = <span>ltrim</span>(<span>$class</span>, '\\'<span>);
        </span><span>//</span><span> 如果 append 为true,就放到 $_events 中名字为 $name 的数组的最后,否则放到最前面</span>
        <span>if</span> (<span>$append</span> || <span>empty</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>])) {
            self</span>::<span>$_events</span>[<span>$name</span>][<span>$class</span>][] = [<span>$handler</span>, <span>$data</span><span>];
        } </span><span>else</span><span> {
            </span><span>array_unshift</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span>], [<span>$handler</span>, <span>$data</span><span>]);
        }
    }

    </span><span>/*</span><span>*
     * Detaches an event handler from a class-level event.
     *
     * This method is the opposite of [[on()]].
     *
     * 移除一个类的事件
     *
     * @param string $class the fully qualified class name from which the event handler needs to be detached.
     * @param string $name the event name.
     * @param callable $handler the event handler to be removed.
     * If it is null, all handlers attached to the named event will be removed.
     * @return boolean whether a handler is found and detached.
     * @see on()
     </span><span>*/</span>
    <span>public</span> <span>static</span> <span>function</span> off(<span>$class</span>, <span>$name</span>, <span>$handler</span> = <span>null</span><span>)
    {
        </span><span>$class</span> = <span>ltrim</span>(<span>$class</span>, '\\'<span>);
        </span><span>if</span> (<span>empty</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>])) {
            </span><span>//</span><span> 不存在该事件</span>
            <span>return</span> <span>false</span><span>;
        }
        </span><span>if</span> (<span>$handler</span> === <span>null</span><span>) {
            </span><span>//</span><span> 如果 handler 为空,直接将在该类下该事件移除,即移出所有的是这个名字的事件</span>
            <span>unset</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>]);
            </span><span>return</span> <span>true</span><span>;
        } </span><span>else</span><span> {
            </span><span>$removed</span> = <span>false</span><span>;
            </span><span>//</span><span> 如果 $handler 不为空,循环 $_events 找到相应的 handler,只移除这个 handler 和 data 组成的数组</span>
            <span>foreach</span> (self::<span>$_events</span>[<span>$name</span>][<span>$class</span>] <span>as</span> <span>$i</span> => <span>$event</span><span>) {
                </span><span>if</span> (<span>$event</span>[0] === <span>$handler</span><span>) {
                    </span><span>unset</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span>][<span>$i</span><span>]);
                    </span><span>$removed</span> = <span>true</span><span>;
                }
            }
            </span><span>if</span> (<span>$removed</span><span>) {
                </span><span>//</span><span> 移除之后,使数组重新变成一个自然数组</span>
                self::<span>$_events</span>[<span>$name</span>][<span>$class</span>] = <span>array_values</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>]);
            }

            </span><span>return</span> <span>$removed</span><span>;
        }
    }

    </span><span>/*</span><span>*
     * Returns a value indicating whether there is any handler attached to the specified class-level event.
     * Note that this method will also check all parent classes to see if there is any handler attached
     * to the named event.
     * 检测在某个类或者对象是否具有某个事件
     * @param string|object $class the object or the fully qualified class name specifying the class-level event.
     * @param string $name the event name.
     * @return boolean whether there is any handler attached to the event.
     </span><span>*/</span>
    <span>public</span> <span>static</span> <span>function</span> hasHandlers(<span>$class</span>, <span>$name</span><span>)
    {
        </span><span>if</span> (<span>empty</span>(self::<span>$_events</span>[<span>$name</span><span>])) {
            </span><span>//</span><span> 不存在,直接返回</span>
            <span>return</span> <span>false</span><span>;
        }
        </span><span>if</span> (<span>is_object</span>(<span>$class</span><span>)) {
            </span><span>//</span><span> 如果是一个 object,就获取其类名</span>
            <span>$class</span> = <span>get_class</span>(<span>$class</span><span>);
        } </span><span>else</span><span> {
            </span><span>//</span><span> 如果是一个类名,就去掉 class 最左边的斜杠</span>
            <span>$class</span> = <span>ltrim</span>(<span>$class</span>, '\\'<span>);
        }
        </span><span>//</span><span> 如果该类中找不到,就去父类中找,直到找到或者没有父类了为止</span>
        <span>do</span><span> {
            </span><span>if</span> (!<span>empty</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>])) {
                </span><span>return</span> <span>true</span><span>;
            }
        } </span><span>while</span> ((<span>$class</span> = <span>get_parent_class</span>(<span>$class</span>)) !== <span>false</span><span>);

        </span><span>return</span> <span>false</span><span>;
    }

    </span><span>/*</span><span>*
     * Triggers a class-level event.
     * This method will cause invocation of event handlers that are attached to the named event
     * for the specified class and all its parent classes.
     * 触发某个类或者对象的某个事件
     * @param string|object $class the object or the fully qualified class name specifying the class-level event.
     * @param string $name the event name.
     * @param Event $event the event parameter. If not set, a default [[Event]] object will be created.
     </span><span>*/</span>
    <span>public</span> <span>static</span> <span>function</span> trigger(<span>$class</span>, <span>$name</span>, <span>$event</span> = <span>null</span><span>)
    {
        </span><span>if</span> (<span>empty</span>(self::<span>$_events</span>[<span>$name</span><span>])) {
            </span><span>return</span><span>;
        }
        </span><span>if</span> (<span>$event</span> === <span>null</span><span>) {
            </span><span>//</span><span> 事件不存在,就创建一个 Event 对象</span>
            <span>$event</span> = <span>new</span> <span>static</span><span>;
        }
        </span><span>//</span><span> 设置event对象的属性,默认是未被处理的</span>
        <span>$event</span>->handled = <span>false</span><span>;
        </span><span>$event</span>->name = <span>$name</span><span>;

        </span><span>if</span> (<span>is_object</span>(<span>$class</span><span>)) {
            </span><span>if</span> (<span>$event</span>->sender === <span>null</span><span>) {
                </span><span>//</span><span> 如果 $class 是个对象,并且是 sender 为空,就将 $class 赋给 sender,即 $class 就是触发事件的对象</span>
                <span>$event</span>->sender = <span>$class</span><span>;
            }
            </span><span>$class</span> = <span>get_class</span>(<span>$class</span><span>);
        } </span><span>else</span><span> {
            </span><span>$class</span> = <span>ltrim</span>(<span>$class</span>, '\\'<span>);
        }
        </span><span>//</span><span> 循环类的 $_event,直到遇到 $event->handled 为真或者没有父类了为止</span>
        <span>do</span><span> {
            </span><span>if</span> (!<span>empty</span>(self::<span>$_events</span>[<span>$name</span>][<span>$class</span><span>])) {
                </span><span>foreach</span> (self::<span>$_events</span>[<span>$name</span>][<span>$class</span>] <span>as</span> <span>$handler</span><span>) {
                    </span><span>//</span><span> 将参数赋到 event 对象的 data 属性上</span>
                    <span>$event</span>->data = <span>$handler</span>[1<span>];
                    </span><span>//</span><span> 调用 $handler 方法
                    // 在方法中,可以用 $this->data 取到相应的参数
                    // 也可以在其中设置 $this->handled 的值,中断后续事件的触发</span>
                    <span>call_user_func</span>(<span>$handler</span>[0], <span>$event</span><span>);
                    </span><span>//</span><span> 当某个 handled 被设置为 true 时,执行到这个事件的时候,会停止,并忽略剩下的事件</span>
                    <span>if</span> (<span>$event</span>-><span>handled) {
                        </span><span>return</span><span>;
                    }
                }
            }
        } </span><span>while</span> ((<span>$class</span> = <span>get_parent_class</span>(<span>$class</span>)) !== <span>false</span><span>);
    }
}</span>
登录后复制

通过上面代码可以看出,类级别的 Event,其本质就是在 Event 类中的 $_events 变量中存储事件,触发事件的时候,只需将其取出,执行即可。

$_events里面的数据结构大概如下:

<span>[
    </span>'add' =><span> [
        </span>'Child' =><span> [
            [</span><span>function</span> (<span>$event</span>) { ... }, <span>$data</span>],<span>
            [[</span><span>$object</span>, 'handleAdd'], <span>null</span>],<span>
            [[</span>'ChildClass', 'handleAdd'], <span>$data</span>],<span>
            [</span>'handleAdd', <span>$data</span><span>]
        ]</span>,
        'ChildClass' =><span> [
            </span>...<span>
        ]
    ]</span>,
    'delete' =><span> [
        </span>...<span>
    ]
]</span>
登录后复制

之后讲到yii\base\Component类时,我们会再来说一下实例级别的事件。

 

 

对 Yii2 源码有兴趣的同学可以关注项目 yii2-2.0.3-annotated,现在在上面已经添加了不少关于 Yii2 源码的注释,之后还会继续添加~

 

有兴趣的同学也可以参与进来,提交 Yii2 源码的注释。

 

 

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/1085573.htmlTechArticleYii2的深入学习--yii\base\Event 类,yii2--yii 根据之前一篇文章,我们知道 Yii2 的事件分两类,一是类级别的事件,二是实例级别的事件。类级别...
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前 By 尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

php如何使用Yii3框架? php如何使用Yii3框架? May 31, 2023 pm 10:42 PM

随着互联网的不断发展,Web应用程序开发的需求也越来越高。对于开发人员而言,开发应用程序需要一个稳定、高效、强大的框架,这样可以提高开发效率。Yii是一款领先的高性能PHP框架,它提供了丰富的特性和良好的性能。Yii3是Yii框架的下一代版本,它在Yii2的基础上进一步优化了性能和代码质量。在这篇文章中,我们将介绍如何使用Yii3框架来开发PHP应用程序。

如何使用PHP框架Yii开发一个高可用的云备份系统 如何使用PHP框架Yii开发一个高可用的云备份系统 Jun 27, 2023 am 09:04 AM

随着云计算技术的不断发展,数据的备份已经成为了每个企业必须要做的事情。在这样的背景下,开发一款高可用的云备份系统尤为重要。而PHP框架Yii是一款功能强大的框架,可以帮助开发者快速构建高性能的Web应用程序。下面将介绍如何使用Yii框架开发一款高可用的云备份系统。设计数据库模型在Yii框架中,数据库模型是非常重要的一部分。因为数据备份系统需要用到很多的表和关

Yii2 vs Phalcon:哪个框架更适合开发显卡渲染应用? Yii2 vs Phalcon:哪个框架更适合开发显卡渲染应用? Jun 19, 2023 am 08:09 AM

在当前信息时代,大数据、人工智能、云计算等技术已经成为了各大企业关注的热点。在这些技术中,显卡渲染技术作为一种高性能图形处理技术,受到了越来越多的关注。显卡渲染技术被广泛应用于游戏开发、影视特效、工程建模等领域。而对于开发者来说,选择一个适合自己项目的框架,是一个非常重要的决策。在当前的语言中,PHP是一种颇具活力的语言,一些优秀的PHP框架如Yii2、Ph

Symfony vs Yii2:哪个框架更适合开发大型Web应用? Symfony vs Yii2:哪个框架更适合开发大型Web应用? Jun 19, 2023 am 10:57 AM

随着Web应用需求的不断增长,开发者们在选择开发框架方面也越来越有选择的余地。Symfony和Yii2是两个备受欢迎的PHP框架,它们都具有强大的功能和性能,但在面对需要开发大型Web应用时,哪个框架更适合呢?接下来我们将对Symphony和Yii2进行比较分析,以帮助你更好地进行选择。基本概述Symphony是一个由PHP编写的开源Web应用框架,它是建立

Yii框架中的数据查询:高效地访问数据 Yii框架中的数据查询:高效地访问数据 Jun 21, 2023 am 11:22 AM

Yii框架是一个开源的PHPWeb应用程序框架,提供了众多的工具和组件,简化了Web应用程序开发的流程,其中数据查询是其中一个重要的组件之一。在Yii框架中,我们可以使用类似SQL的语法来访问数据库,从而高效地查询和操作数据。Yii框架的查询构建器主要包括以下几种类型:ActiveRecord查询、QueryBuilder查询、命令查询和原始SQL查询

yii如何将对象转化为数组或直接输出为json格式 yii如何将对象转化为数组或直接输出为json格式 Jan 08, 2021 am 10:13 AM

yii框架:本文为大家介绍了yii将对象转化为数组或直接输出为json格式的方法,具有一定的参考价值,希望能够帮助到大家。

Yii2编程指南:运行Cron服务的方法 Yii2编程指南:运行Cron服务的方法 Sep 01, 2023 pm 11:21 PM

如果您问“Yii是什么?”查看我之前的教程:Yii框架简介,其中回顾了Yii的优点,并概述了2014年10月发布的Yii2.0的新增功能。嗯>在这个使用Yii2编程系列中,我将指导读者使用Yii2PHP框架。在今天的教程中,我将与您分享如何利用Yii的控制台功能来运行cron作业。过去,我在cron作业中使用了wget—可通过Web访问的URL来运行我的后台任务。这引发了安全问题并存在一些性能问题。虽然我在我们的启动系列安全性专题中讨论了一些减轻风险的方法,但我曾希望过渡到控制台驱动的命令

Yii框架中的表单构建器:构建复杂表单 Yii框架中的表单构建器:构建复杂表单 Jun 21, 2023 am 10:09 AM

随着互联网的快速发展,Web应用越来越成为人们生活中不可或缺的一部分。而表单是Web应用中不可或缺的元素之一,其用于收集用户数据,让Web应用能够更好地为用户服务。Yii框架是一个快速、高效、灵活的PHP框架,可以帮助开发人员更加快速地开发Web应用。Yii框架中的表单构建器(FormBuilder)可以让开发人员轻松地构建复杂的表单,让Web应用具有更好

See all articles