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 源码的注释。

PHP和Python各有优势,选择应基于项目需求。1.PHP适合web开发,语法简单,执行效率高。2.Python适用于数据科学和机器学习,语法简洁,库丰富。

PHP不是在消亡,而是在不断适应和进化。1)PHP从1994年起经历多次版本迭代,适应新技术趋势。2)目前广泛应用于电子商务、内容管理系统等领域。3)PHP8引入JIT编译器等功能,提升性能和现代化。4)使用OPcache和遵循PSR-12标准可优化性能和代码质量。

PHP的未来将通过适应新技术趋势和引入创新特性来实现:1)适应云计算、容器化和微服务架构,支持Docker和Kubernetes;2)引入JIT编译器和枚举类型,提升性能和数据处理效率;3)持续优化性能和推广最佳实践。

在PHP中,trait适用于需要方法复用但不适合使用继承的情况。1)trait允许在类中复用方法,避免多重继承复杂性。2)使用trait时需注意方法冲突,可通过insteadof和as关键字解决。3)应避免过度使用trait,保持其单一职责,以优化性能和提高代码可维护性。

依赖注入容器(DIC)是一种管理和提供对象依赖关系的工具,用于PHP项目中。DIC的主要好处包括:1.解耦,使组件独立,代码易维护和测试;2.灵活性,易替换或修改依赖关系;3.可测试性,方便注入mock对象进行单元测试。

SplFixedArray在PHP中是一种固定大小的数组,适用于需要高性能和低内存使用量的场景。1)它在创建时需指定大小,避免动态调整带来的开销。2)基于C语言数组,直接操作内存,访问速度快。3)适合大规模数据处理和内存敏感环境,但需谨慎使用,因其大小固定。

PHP通过$\_FILES变量处理文件上传,确保安全性的方法包括:1.检查上传错误,2.验证文件类型和大小,3.防止文件覆盖,4.移动文件到永久存储位置。

JavaScript中处理空值可以使用NullCoalescingOperator(??)和NullCoalescingAssignmentOperator(??=)。1.??返回第一个非null或非undefined的操作数。2.??=将变量赋值为右操作数的值,但前提是该变量为null或undefined。这些操作符简化了代码逻辑,提高了可读性和性能。


热AI工具

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

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

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

EditPlus 中文破解版
体积小,语法高亮,不支持代码提示功能

SublimeText3 Linux新版
SublimeText3 Linux最新版

WebStorm Mac版
好用的JavaScript开发工具

禅工作室 13.0.1
功能强大的PHP集成开发环境

Atom编辑器mac版下载
最流行的的开源编辑器