Maison >développement back-end >tutoriel php >Phalcon自动加载(PHP自动加载),phalcon加载php_PHP教程
通过 include() 或 require() 函数,可以在PHP程序执行之前在该文件中插入一个文件的内容。
<p>区别:<span>处理错误的方式不同</span>。<strong>include() 函数</strong>会生成一个<span><strong>警告</strong></span>(但是脚本会继续执行),而 <strong>require()</strong> 函数会生成一个<span><strong>致命错误</strong></span>(fatal error)(在错误发生后脚本会停止执行)</p>
* 正因为在文件不存在或被重命名后脚本不会继续执行,因此我们推荐使用 require() 而不是 include()。
参考文章:php手册 和 PHP的类自动加载机制
在php5之前,各php框架实现类的加载,一般要按照某种约定实现一个遍历目录,自动加载符合约定条件的文件类或函数。因此在php5之前类的使用并没有现在频繁。
在php5之后,当加载php类的时候,如果类所在文件夹并没有被包含进来或是类名出错时,Zend引擎会自动调用__autoload函数。__autoload函数需要用户自己实现。
在php5.1.2版本之后,可以使用spl_autoload_register函数自定义加载处理函数。当没有调用此函数,默认情况下会使用spl自定义的spl_autoload函数。
<span>function</span> __autoload(<span>$className</span><span>) { </span><span>$file</span> = <span>$className</span> . '.php'<span>; </span><span>if</span> (<span>is_file</span>(<span>$file</span><span>)) { </span><span>require</span>(<span>$file</span><span>); }</span><span>else</span><span>{ </span><span>echo</span> 'no this ' . <span>$className</span> . ' class file'<span>; } } </span><span>$demo</span> = <span>new</span> Demo();
事实上,我们可以看到__autoload
至少需要做三件事(“三步走”),它们分别是:
在第一步和第二步中,我们必须约定类名与文件的映射方法,只有这样我们才能够依据类名找到其所对应的文件,实现加载。
因此__autoload自动加载中,最重要的就是指定类名与其所在文件的对应关系。当有大量的类需要包含进来的时候,我们只需要确立相应的规则,然后将类名与其对应的文件进行映射,就能够实现惰性加载(lazy loading)了。
<p><strong>Tip:</strong>spl_autoload_register() 提供了一种更加灵活的方式来实现类的自动加载。因此,不再建议使用 __autoload() 函数,在以后的版本中它可能被弃用。</p>
引言:如果在一个php系统实现中,使用了很多的其他类库,这些类库可能是由不同的工程师进行开发的,因此类名与其所在文件的映射规则不尽相同。这时候如果要实现类库的自动加载,就必须在__autoload函数中将所有的映射规则全部实现。这就会导致__autoload会非常复杂,甚至无法实现。同时还会使得__autoload函数十分臃肿。为将来系统的维护和性能带来很大的负面影响。(__autoload的弊端)
注册给定的函数作为__autoload的实现。简单来说就是将函数注册之SPL的__autoload函数栈中,并移除系统默认的__autload()函数。
<span>function</span> __autoload(<span>$className</span><span>) { </span><span>echo</span> 'autload class:', <span>$className</span>, '<br />'<span>; } </span><span>function</span> classLoader(<span>$className</span><span>) { </span><span>echo</span> 'SPL load class:', <span>$className</span>, '<br />'<span>; } spl_autoload_register(</span>'classLoader'<span>); </span><span>new</span> Test();<span>//</span><span>结果:SPL load class:Test </span>
<p><strong>Tip:</strong></p>
<pre class="code"><span><strong><code>bool spl_autoload_register ([ callable $autoload_function [, bool $throw = true [, bool $prepend = false ]]] ) </code></strong></span>
<p><strong>备注:</strong>SPL自动加载功能是由spl_autoload() ,spl_autoload_register(), spl_autoload_functions() ,spl_autoload_extensions()和spl_autoload_call()函数提供的。</p>
Phalcon\Loader 通用类加载器(Universal Class Loader),意在根据协议帮助项目自动加载项目中的类(This component helps to load your project classes automatically based on some conventions)。Phalcon支持四种类加载方式,先后顺序分别是注册类名、注册命名空间、注册前缀和注册文件夹的方式。
Phalcon的默认文件后缀为php,当然你自己也可以配置(setExtensions())。
<?<span>php </span><span>$loader</span> = <span>new</span><span> \Phalcon\Loader(); </span><span>$loader</span>-><span>registerClasses( </span><span>array</span><span>( </span>"Some" => "library/OtherComponent/Other/Some.php", "Example\Base" => "vendor/example/adapters/Example/BaseClass.php",<span> ) ); </span><span>$loader</span>-><span>register(); </span><span>//</span><span> i.e. library/OtherComponent/Other/Some.php</span> <span>$some</span> = <span>new</span> Some();
<p>具体实现:</p>
<?<span>php </span><span>$loader</span> = <span>new</span><span> \Phalcon\Loader(); </span><span>$loader</span>-><span>registerNamespaces( </span><span>array</span><span>( </span>"Example\Base" => "vendor/example/base/", "Example\Adapter" => "vendor/example/adapter/", "Example" => "vendor/example/",<span> ) ); </span><span>$loader</span>-><span>register(); </span><span>//</span><span> vendor/example/adapter/Some.php</span> <span>$some</span> = <span>new</span> Example\Adapter\Some();
使用命名空间或外部库组织代码时,你可以利用注册命名空间的方式来自动加载其包含的库。
对于命名空间对应的路径,要其末尾加一个斜杠。
<p>具体实现:</p>
<?<span>php </span><span>$loader</span> = <span>new</span><span> \Phalcon\Loader(); </span><span>$loader</span>-><span>registerPrefixes( </span><span>array</span><span>( </span>"Example_Base" => "vendor/example/base/", "Example_Adapter" => "vendor/example/adapter/", "Example_" => "vendor/example/",<span> ) ); </span><span>$loader</span>-><span>register(); </span><span>//</span><span> vendor/example/adapter/Some.php</span> <span>$some</span> = <span>new</span> Example_Adapter_Some();
类似于命名空间,从2.1.0开始phalcon将不再支持前缀。
<p>具体实现:</p>
<?<span>php </span><span>$loader</span> = <span>new</span><span> \Phalcon\Loader(); </span><span>$loader</span>-><span>registerDirs( </span><span>array</span><span>( </span>"library/MyComponent/", "library/OtherComponent/Other/", "vendor/example/adapters/", "vendor/example/"<span> ) ); </span><span>$loader</span>-><span>register(); </span><span>//</span><span> i.e. library/OtherComponent/Other/Some.php</span> <span>$some</span> = <span>new</span> Some();
可以自动加载注册目录下的类文件。但是该方法在性能方面并不被推荐,因为Phalcon将在个文件夹下大量查找与类名相同的文件。在使用注册目录自动加载时,要注意注册目录的相关性,即将重要的目录放在前面。
<p>具体实现:</p>
即为当前自动加载数据添加额外的值。
<?<span>php </span><span>//</span><span> Adding more directories</span> <span>$loader</span>-><span>registerDirs( </span><span>array</span><span>( </span>"../app/library/", "../app/plugins/"<span> )</span>, <span>true</span><span> );</span>
注册时添加第二个参数值true,使其与原数组合并。
没有进行任何安全检查的自动加载器,如下:
<?<span>php </span><span>//</span><span>Basic autoloader</span> spl_autoload_register(<span>function</span>(<span>$className</span><span>) { </span><span>if</span> (<span>file_exists</span>(<span>$className</span> . '.php'<span>)) { </span><span>require</span> <span>$className</span> . '.php'<span>; } });</span>
假如我们没有进行任何安全检查时,如果误启了自动加载器,那么恶意准备的字符串就回作为参数访问程序中的重要文件。
<?<span>php </span><span>//</span><span>This variable is not filtered and comes from an insecure source</span> <span>$className</span> = '../processes/important-process'<span>; </span><span>//</span><span>Check if the class exists triggering the auto-loader</span> <span>if</span> (<span>class_exists</span>(<span>$className</span><span>)) { </span><span>//</span><span>...</span> }
Phalcon的做法是删除任何无用的字符串,减少被攻击的可能性。
在下面的例子中,而不必使用类加载器,使我们获得调试信息的流程操作:
<?<span>php </span><span>$eventsManager</span> = <span>new</span><span> \Phalcon\Events\Manager(); </span><span>$loader</span> = <span>new</span><span> \Phalcon\Loader(); </span><span>$loader</span>->registerNamespaces(<span>array</span><span>( </span>'Example\\Base' => 'vendor/example/base/', 'Example\\Adapter' => 'vendor/example/adapter/', 'Example' => 'vendor/example/'<span> )); </span><span>//</span><span>Listen all the loader events</span> <span>$eventsManager</span>->attach('loader', <span>function</span>(<span>$event</span>, <span>$loader</span><span>) { </span><span>if</span> (<span>$event</span>-><span>getType</span>() == 'beforeCheckPath'<span>) { </span><span>echo</span> <span>$loader</span>-><span>getCheckedPath(); } }); </span><span>$loader</span>->setEventsManager(<span>$eventsManager</span><span>); </span><span>$loader</span>->register();
Phalcon自动加载支持以下事件: