首页 后端开发 php教程 yii源码分析四——非核心类的导入注册

yii源码分析四——非核心类的导入注册

Jun 13, 2016 pm 12:17 PM
alias config include self

yii源码分析4——非核心类的导入注册

转载请注明: TheViper http://www.cnblogs.com/TheViper 

在yii源码分析1中说到spl_autoload_register注册给定的函数作为 __autoload 的实现,在这里是autoload().

<span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">function</span> autoload(<span style="color: #800080;">$className</span><span style="color: #000000;">) {        </span><span style="color: #0000ff;">include</span> self::<span style="color: #800080;">$_coreClasses</span> [<span style="color: #800080;">$className</span><span style="color: #000000;">];     }</span>
登录后复制

实际上这个autoload()是没有考虑非核心文件的引入的。比如,在app文件夹经常会有自定义的一些重要文件夹,比如'application.utils.*(工具类),'application.filters.*'(过滤类),'application.validators.*'(校验类)等。

在实际用的时候,是不用一个一个include的,直接new就可以了,yii已经帮我们做了include的工作。而这个工作就是在autoload()里面做的。

上面的代码很显然没有考虑非核心文件的引入,这是我的疏忽。

那yii是怎么帮我们引入非核心文件的?

这要从CApplication说起。

<span style="color: #0000ff;">abstract</span> <span style="color: #0000ff;">class</span> CApplication <span style="color: #0000ff;">extends</span><span style="color: #000000;"> CModule {    </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> __construct(<span style="color: #800080;">$config</span> = <span style="color: #0000ff;">null</span><span style="color: #000000;">) {        </span><span style="color: #0000ff;">if</span> (<span style="color: #008080;">is_string</span> ( <span style="color: #800080;">$config</span><span style="color: #000000;"> ))            </span><span style="color: #800080;">$config</span> = <span style="color: #0000ff;">require</span> (<span style="color: #800080;">$config</span><span style="color: #000000;">);        Yii</span>::setApplication ( <span style="color: #800080;">$this</span> );<span style="color: #008000;">//</span><span style="color: #008000;">保存整个app实例</span>        <span style="color: #0000ff;">if</span> (<span style="color: #0000ff;">isset</span> ( <span style="color: #800080;">$config</span> ['basePath'<span style="color: #000000;">] )) {            </span><span style="color: #800080;">$this</span>->setBasePath ( <span style="color: #800080;">$config</span> ['basePath'<span style="color: #000000;">] );            </span><span style="color: #0000ff;">unset</span> ( <span style="color: #800080;">$config</span> ['basePath'<span style="color: #000000;">] );        } </span><span style="color: #0000ff;">else</span>            <span style="color: #800080;">$this</span>->setBasePath ( 'protected'<span style="color: #000000;"> );        </span><span style="color: #008000;">//</span><span style="color: #008000;">设置别名,后面就可以用application表示basePath了</span>        Yii::setPathOfAlias ( 'application', <span style="color: #800080;">$this</span>-><span style="color: #000000;">getBasePath () );        </span><span style="color: #008000;">//</span><span style="color: #008000;">钩子,模块 预 初始化时执行,子类实现。不过这时,配置还没有写入框架</span>        <span style="color: #800080;">$this</span>-><span style="color: #000000;">preinit ();        </span><span style="color: #800080;">$this</span>-><span style="color: #000000;">registerCoreComponents ();        </span><span style="color: #008000;">//</span><span style="color: #008000;">父类实现</span>        <span style="color: #800080;">$this</span>->configure ( <span style="color: #800080;">$config</span><span style="color: #000000;"> );        </span><span style="color: #008000;">//</span><span style="color: #008000;">加载静态应用组件</span>        <span style="color: #800080;">$this</span>-><span style="color: #000000;">preloadComponents ();        </span><span style="color: #008000;">//</span><span style="color: #008000;">这才开始初始化模块</span>        <span style="color: #800080;">$this</span>-><span style="color: #000000;">init ();    }</span>
登录后复制

注意到里面的$this->configure ( $config );,$config是传入的配置文件,是一个数组,非核心文件的定义就是在这里面,比如引入工具类文件夹

<span style="color: #000000;">php</span><span style="color: #0000ff;">return</span> <span style="color: #0000ff;">array</span><span style="color: #000000;"> (    </span>'basePath' => <span style="color: #008080;">dirname</span> ( <span style="color: #ff00ff;">__FILE__</span> ) . DIRECTORY_SEPARATOR . '..',    'import' => <span style="color: #0000ff;">array</span><span style="color: #000000;"> (        </span>'application.utils.*'<span style="color: #000000;">    )    );</span>?> 
登录后复制

然后在父类CModule

    <span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> configure(<span style="color: #800080;">$config</span><span style="color: #000000;">) {        </span><span style="color: #0000ff;">if</span> (<span style="color: #008080;">is_array</span> ( <span style="color: #800080;">$config</span><span style="color: #000000;"> )) {            </span><span style="color: #0000ff;">foreach</span> ( <span style="color: #800080;">$config</span> <span style="color: #0000ff;">as</span> <span style="color: #800080;">$key</span> => <span style="color: #800080;">$value</span><span style="color: #000000;"> )                </span><span style="color: #800080;">$this</span>-><span style="color: #800080;">$key</span> = <span style="color: #800080;">$value</span><span style="color: #000000;">;        }    }</span>
登录后复制

这里yii很"狡猾",它在CModule的父类CComponent中重写了__set()

    <span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> __set(<span style="color: #800080;">$name</span>,<span style="color: #800080;">$value</span><span style="color: #000000;">)    {        </span><span style="color: #800080;">$setter</span>='set'.<span style="color: #800080;">$name</span><span style="color: #000000;">;        </span><span style="color: #0000ff;">if</span>(<span style="color: #008080;">method_exists</span>(<span style="color: #800080;">$this</span>,<span style="color: #800080;">$setter</span><span style="color: #000000;">))            </span><span style="color: #0000ff;">return</span> <span style="color: #800080;">$this</span>-><span style="color: #800080;">$setter</span>(<span style="color: #800080;">$value</span><span style="color: #000000;">);        </span><span style="color: #0000ff;">else</span>....<span style="color: #000000;">    }</span>
登录后复制

可以看到,如果CModule中如果有设置yii指定参数(比如import)的方法,就会调用它,而我之前裁剪的时候,把CModule中的setImport()删掉了。

另外可以看到basePath, params, modules, import, components 是yii保留的参数名。

    <span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> setImport(<span style="color: #800080;">$aliases</span><span style="color: #000000;">)    {        </span><span style="color: #0000ff;">foreach</span>(<span style="color: #800080;">$aliases</span> <span style="color: #0000ff;">as</span> <span style="color: #800080;">$alias</span><span style="color: #000000;">)            Yii</span>::import(<span style="color: #800080;">$alias</span><span style="color: #000000;">);    }</span>
登录后复制

然后是YiiBase里面的import()

    <span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">function</span> import(<span style="color: #800080;">$alias</span>, <span style="color: #800080;">$forceInclude</span> = <span style="color: #0000ff;">false</span><span style="color: #000000;">) {        </span><span style="color: #0000ff;">if</span> (<span style="color: #0000ff;">isset</span> ( self::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] )) <span style="color: #008000;">//</span><span style="color: #008000;">是否已经存在路径</span>            <span style="color: #0000ff;">return</span> self::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span><span style="color: #000000;">];                </span><span style="color: #0000ff;">if</span> (<span style="color: #008080;">class_exists</span> ( <span style="color: #800080;">$alias</span>, <span style="color: #0000ff;">false</span> ) || <span style="color: #008080;">interface_exists</span> ( <span style="color: #800080;">$alias</span>, <span style="color: #0000ff;">false</span> ))<span style="color: #008000;">//</span><span style="color: #008000;">类是否已经定义,针对如urlManager这样的已定义于$_coreClasses[]的类</span>            <span style="color: #0000ff;">return</span> self::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] = <span style="color: #800080;">$alias</span><span style="color: #000000;">;        </span><span style="color: #0000ff;">if</span> ((<span style="color: #800080;">$pos</span> = <span style="color: #008080;">strrpos</span> ( <span style="color: #800080;">$alias</span>, '.' )) === <span style="color: #0000ff;">false</span>)         <span style="color: #008000;">//</span><span style="color: #008000;">直接是文件名</span><span style="color: #000000;">        {            </span><span style="color: #008000;">//</span><span style="color: #008000;"> try to autoload the class with an autoloader if $forceInclude is true</span>            <span style="color: #0000ff;">if</span> (<span style="color: #800080;">$forceInclude</span> && (Yii::autoload ( <span style="color: #800080;">$alias</span>, <span style="color: #0000ff;">true</span> ) || <span style="color: #008080;">class_exists</span> ( <span style="color: #800080;">$alias</span>, <span style="color: #0000ff;">true</span><span style="color: #000000;"> )))                self</span>::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] = <span style="color: #800080;">$alias</span><span style="color: #000000;">;            </span><span style="color: #0000ff;">return</span> <span style="color: #800080;">$alias</span><span style="color: #000000;">;        }                </span><span style="color: #800080;">$className</span> = ( <span style="color: #0000ff;">string</span> ) <span style="color: #008080;">substr</span> ( <span style="color: #800080;">$alias</span>, <span style="color: #800080;">$pos</span> + 1<span style="color: #000000;"> );        </span><span style="color: #800080;">$isClass</span> = <span style="color: #800080;">$className</span> !== '*'<span style="color: #000000;">;        </span><span style="color: #008000;">//</span><span style="color: #008000;">是否为路径+类名</span>        <span style="color: #0000ff;">if</span> (<span style="color: #800080;">$isClass</span> && (<span style="color: #008080;">class_exists</span> ( <span style="color: #800080;">$className</span>, <span style="color: #0000ff;">false</span> ) || <span style="color: #008080;">interface_exists</span> ( <span style="color: #800080;">$className</span>, <span style="color: #0000ff;">false</span><span style="color: #000000;"> )))            </span><span style="color: #0000ff;">return</span> self::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] = <span style="color: #800080;">$className</span><span style="color: #000000;">;        </span><span style="color: #008000;">//</span><span style="color: #008000;">获取真实路径</span>        <span style="color: #0000ff;">if</span> ((<span style="color: #800080;">$path</span> = self::getPathOfAlias ( <span style="color: #800080;">$alias</span> )) !== <span style="color: #0000ff;">false</span><span style="color: #000000;">) {            </span><span style="color: #008000;">//</span><span style="color: #008000;">是否以*结尾,如application.utils.*</span>            <span style="color: #0000ff;">if</span> (<span style="color: #800080;">$isClass</span><span style="color: #000000;">) {                </span><span style="color: #0000ff;">if</span> (<span style="color: #800080;">$forceInclude</span><span style="color: #000000;">) {                    </span><span style="color: #0000ff;">if</span> (<span style="color: #008080;">is_file</span> ( <span style="color: #800080;">$path</span> . '.php'<span style="color: #000000;"> ))                        </span><span style="color: #0000ff;">require</span> (<span style="color: #800080;">$path</span> . '.php'<span style="color: #000000;">);                    </span><span style="color: #0000ff;">else</span>                        <span style="color: #0000ff;">throw</span> <span style="color: #0000ff;">new</span> CException ( Yii::t ( 'yii', 'Alias "{alias}" is invalid. Make sure it points to an existing PHP file and the file is readable.', <span style="color: #0000ff;">array</span><span style="color: #000000;"> (                                </span>'{alias}' => <span style="color: #800080;">$alias</span><span style="color: #000000;">                         ) ) );                    self</span>::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] = <span style="color: #800080;">$className</span><span style="color: #000000;">;                } </span><span style="color: #0000ff;">else</span><span style="color: #000000;">                    self</span>::<span style="color: #800080;">$classMap</span> [<span style="color: #800080;">$className</span>] = <span style="color: #800080;">$path</span> . '.php'<span style="color: #000000;">;                </span><span style="color: #0000ff;">return</span> <span style="color: #800080;">$className</span><span style="color: #000000;">;            } </span><span style="color: #0000ff;">else</span>             <span style="color: #008000;">//</span><span style="color: #008000;"> a directory</span><span style="color: #000000;">            {                </span><span style="color: #0000ff;">if</span> (self::<span style="color: #800080;">$_includePaths</span> === <span style="color: #0000ff;">null</span><span style="color: #000000;">) {                    self</span>::<span style="color: #800080;">$_includePaths</span> = <span style="color: #008080;">array_unique</span> ( <span style="color: #008080;">explode</span> ( PATH_SEPARATOR, <span style="color: #008080;">get_include_path</span><span style="color: #000000;"> () ) );                    </span><span style="color: #0000ff;">if</span> ((<span style="color: #800080;">$pos</span> = <span style="color: #008080;">array_search</span> ( '.', self::<span style="color: #800080;">$_includePaths</span>, <span style="color: #0000ff;">true</span> )) !== <span style="color: #0000ff;">false</span><span style="color: #000000;">)                        </span><span style="color: #0000ff;">unset</span> ( self::<span style="color: #800080;">$_includePaths</span> [<span style="color: #800080;">$pos</span><span style="color: #000000;">] );                }                                </span><span style="color: #008080;">array_unshift</span> ( self::<span style="color: #800080;">$_includePaths</span>, <span style="color: #800080;">$path</span><span style="color: #000000;"> );                                </span><span style="color: #0000ff;">if</span> (self::<span style="color: #800080;">$enableIncludePath</span> && <span style="color: #008080;">set_include_path</span> ( '.' . PATH_SEPARATOR . <span style="color: #008080;">implode</span> ( PATH_SEPARATOR, self::<span style="color: #800080;">$_includePaths</span> ) ) === <span style="color: #0000ff;">false</span><span style="color: #000000;">)                    self</span>::<span style="color: #800080;">$enableIncludePath</span> = <span style="color: #0000ff;">false</span><span style="color: #000000;">;                </span><span style="color: #0000ff;">return</span> self::<span style="color: #800080;">$_imports</span> [<span style="color: #800080;">$alias</span>] = <span style="color: #800080;">$path</span><span style="color: #000000;">;            }        }    }</span>
登录后复制

 

一系列的判断,最后走到最后的else,将path写入到$_imports,这时仍然没有include.

include在autoload()

    <span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">function</span> autoload(<span style="color: #800080;">$className</span><span style="color: #000000;">)    {        </span><span style="color: #008000;">//</span><span style="color: #008000;"> use include so that the error PHP file may appear</span>        <span style="color: #0000ff;">if</span>(<span style="color: #0000ff;">isset</span>(self::<span style="color: #800080;">$classMap</span>[<span style="color: #800080;">$className</span><span style="color: #000000;">]))            </span><span style="color: #0000ff;">include</span>(self::<span style="color: #800080;">$classMap</span>[<span style="color: #800080;">$className</span><span style="color: #000000;">]);        </span><span style="color: #0000ff;">elseif</span>(<span style="color: #0000ff;">isset</span>(self::<span style="color: #800080;">$_coreClasses</span>[<span style="color: #800080;">$className</span><span style="color: #000000;">]))            </span><span style="color: #0000ff;">include</span>(self::<span style="color: #800080;">$_coreClasses</span>[<span style="color: #800080;">$className</span><span style="color: #000000;">]);        </span><span style="color: #0000ff;">else</span><span style="color: #000000;">        {            </span><span style="color: #008000;">//</span><span style="color: #008000;"> include class file relying on include_path</span>            <span style="color: #0000ff;">if</span>(<span style="color: #008080;">strpos</span>(<span style="color: #800080;">$className</span>,'\\')===<span style="color: #0000ff;">false</span>)  <span style="color: #008000;">//</span><span style="color: #008000;"> class without namespace</span><span style="color: #000000;">            {                </span><span style="color: #0000ff;">if</span>(self::<span style="color: #800080;">$enableIncludePath</span>===<span style="color: #0000ff;">false</span><span style="color: #000000;">)                {                    </span><span style="color: #0000ff;">foreach</span>(self::<span style="color: #800080;">$_includePaths</span> <span style="color: #0000ff;">as</span> <span style="color: #800080;">$path</span><span style="color: #000000;">)                    {                        </span><span style="color: #800080;">$classFile</span>=<span style="color: #800080;">$path</span>.DIRECTORY_SEPARATOR.<span style="color: #800080;">$className</span>.'.php'<span style="color: #000000;">;                        </span><span style="color: #0000ff;">if</span>(<span style="color: #008080;">is_file</span>(<span style="color: #800080;">$classFile</span><span style="color: #000000;">))                        {                            </span><span style="color: #0000ff;">include</span>(<span style="color: #800080;">$classFile</span><span style="color: #000000;">);                            </span><span style="color: #0000ff;">break</span><span style="color: #000000;">;                        }                    }                }                </span><span style="color: #0000ff;">else</span>                    <span style="color: #0000ff;">include</span>(<span style="color: #800080;">$className</span>.'.php'<span style="color: #000000;">);            }            </span><span style="color: #0000ff;">return</span> <span style="color: #008080;">class_exists</span>(<span style="color: #800080;">$className</span>,<span style="color: #0000ff;">false</span>) || <span style="color: #008080;">interface_exists</span>(<span style="color: #800080;">$className</span>,<span style="color: #0000ff;">false</span><span style="color: #000000;">);        }        </span><span style="color: #0000ff;">return</span> <span style="color: #0000ff;">true</span><span style="color: #000000;">;    }</span>
登录后复制

 

如果需要include的是非核心文件,那这里的$className只是一个alias,即文件名的前缀。

裁剪的yii http://files.cnblogs.com/TheViper/framework.zip

如果您觉得本文的内容对您有所帮助,您可以打赏我:

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系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脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

<🎜>:泡泡胶模拟器无穷大 - 如何获取和使用皇家钥匙
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆树的耳语 - 如何解锁抓钩
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系统,解释
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)

热门话题

Java教程
1667
14
CakePHP 教程
1426
52
Laravel 教程
1328
25
PHP教程
1273
29
C# 教程
1255
24
为什么NameResolutionError(self.host, self, e) from e,怎么解决 为什么NameResolutionError(self.host, self, e) from e,怎么解决 Mar 01, 2024 pm 01:20 PM

报错的原因NameResolutionError(self.host,self,e)frome是由urllib3库中的异常类型,这个错误的原因是DNS解析失败,也就是说,试图解析的主机名或IP地址无法找到。这可能是由于输入的URL地址不正确,或者DNS服务器暂时不可用导致的。如何解决解决此错误的方法可能有以下几种:检查输入的URL地址是否正确,确保它是可访问的确保DNS服务器可用,您可以尝试在命令行中使用"ping"命令来测试DNS服务器是否可用尝试使用IP地址而不是主机名来访问网站如果是在代理

Nginx配置中指令root和alias的区别是什么 Nginx配置中指令root和alias的区别是什么 May 12, 2023 pm 12:16 PM

root和alias都可以定义在location模块中,都是用来指定请求资源的真实路径,比如:location/i/{root/data/w3;}请求http://foofish.net/i/top.gif这个地址时,那么在服务器里面对应的真正的资源是/data/w3/i/top.gif文件注意:真实的路径是root指定的值加上location指定的值。而alias正如其名,alias指定的路径是location的别名,不管location的值怎么写,资源的真实路径都是alias指定的路径,比如

如何修复 .NET 解析器返回的错误 0xC00CE556 如何修复 .NET 解析器返回的错误 0xC00CE556 Apr 25, 2023 am 08:34 AM

在安装应用程序的新版本时,Windows可能会显示此错误消息“解析C:\\Windows\Microsoft.Net\Framework\v2.0.50727\Config\machine.configParser返回错误0xC00CE556时出错”。当您的系统启动时,这个问题也会出现。无论您在何种情况下遇到此问题,.NETFramework都是幕后真正的罪魁祸首。您可以使用一些非常简单的修复程序来阻止此错误代码再次出现。修复1–替换损坏的文件您可以轻松地从原始目录中替换损坏的ma

php include和include_once有什么区别 php include和include_once有什么区别 Mar 22, 2023 am 10:38 AM

当我们在使用 PHP 编写网页时,有时我们需要在当前 PHP 文件中包含其他 PHP 文件中的代码。这时,就可以使用 include 或 include_once 函数来实现文件包含。那么,include 和 include_once 到底有什么区别呢?

php如何使用ThinkPHP\Config进行配置管理? php如何使用ThinkPHP\Config进行配置管理? May 31, 2023 pm 02:31 PM

随着PHP语言的不断发展,做为PHP后端框架中广泛使用的ThinkPHP也在不断完善。随着业务场景的逐渐复杂,ThinkPHP中对于配置管理的需求也越来越大。在这种背景下,ThinkPHP提供了丰富的配置管理功能,今天我们就来介绍一下如何通过ThinkPHPConfig实现配置管理。一、ThinkPHPConfig的介绍ThinkPHPConfig是Thin

Python中的self怎么使用 Python中的self怎么使用 May 17, 2023 pm 10:40 PM

在介绍Python的self用法之前,先来介绍下Python中的类和实例我们知道,面向对象最重要的概念就是类(class)和实例(instance),类是抽象的模板,比如学生这个抽象的事物,可以用一个Student类来表示。而实例是根据类创建出来的一个个具体的“对象”,每一个对象都从类中继承有相同的方法,但各自的数据可能不同。1、以Student类为例,在Python中,定义类如下:classStudent(object):pass(Object)表示该类从哪个类继承下来的,Object类是所有

php如何使用CodeIgniter\Config进行配置管理? php如何使用CodeIgniter\Config进行配置管理? Jun 02, 2023 pm 06:01 PM

一、CodeIgniter简介CodeIgniter是一个轻量级且全面的PHP开发框架,旨在为Web开发人员提供快速且强大的工具来构建Web应用程序。它是一个开源的框架,使用MVC架构模式来实现快速开发和基础功能,同时支持多种数据库。二、Config库简介Config库是CodeIgniter框架中的一个组件,用于对代码进行配置管理。Config库包含了很多

nginx,apache的alias和认证功能实例分析 nginx,apache的alias和认证功能实例分析 May 24, 2023 pm 11:10 PM

首先看下看下apache别名怎么配置的:复制代码代码如下:documentroot/www/jb51.net/www这是虚拟主机的根目录吧,但是phpmyadmin不在这个目录下,想访问。servernamewww.jb51.netserveraliasjb51.netalias/sdb"/www/public/phpmyadmin/"就需要别名功能,://www.jb51.net/sdb这样就安全多了。optionsindexesfollowsymlinksallowove

See all articles