PHP的类自动加载机制的笔记

在php中自动加载我们会使用到方法__autoload spl_autoload_register来实现,像我们有几百个类时我们希望是希望使用__autoload来创建,然后直接创建类名就可以了,下面简单的总结了一下自动加载类的用法,希望对各位有帮助.

在PHP5之前,各个PHP框架如果要实现类的自动加载,一般都是按照某种约定自己实现一个遍历目录,自动加载所有符合约定规则的文件的类或函数,当然,PHP5之前对面向对象的支持并不是太好,类的使用也没有现在频繁,在PHP5后,当加载PHP类时,如果类所在文件没有被包含进来,或者类名出错,Zend引擎会自动调用__autoload 函数,此函数需要用户自己实现__autoload函数, 在PHP5.1.2版本后,可以使用spl_autoload_register函数自定义自动加载处理函数,当没有调用此函数,默认情况下会使用SPL自定义的spl_autoload函数,看下面两个例子:

1、__autoload示例:

  1. function __autoload($class_name) {
  2. echo '__autload class:', $class_name, '<br />';
  3. }
  4. new Demo();

以上的代码在最后会输出:__autload class:Demo。

并在此之后报错显示:Fatal error: Class 'Demo' not found

2、spl_autoload_register示例:

  1. function classLoader($class_name) {
  2. echo 'SPL load class:', $class_name, '<br />';
  3. }
  4. spl_autoload_register('classLoader');
  5. new Demo();

以上的代码在最后会输出:SPL load class:Demo。

并在此之后报错显示:Fatal error: Class 'Demo' not found

以上的两个示例表明:当类不存在时(即需要的类不在类符号表),Zend引擎会将再调用一次用户定义的函数,如__autoload或spl_autoload_register注册的函数,如果这两个方法同时存在,那么程序会调用哪一个呢?还是说两个都调用?看下面一个示例,你觉得会输出什么呢?代码如下:

  1. function __autoload($class_name) {
  2. echo '__autload class:', $class_name, '<br />';
  3. }
  4. function classLoader($class_name) {
  5. echo 'SPL load class:', $class_name, '<br />';
  6. }
  7. spl_autoload_register('classLoader');
  8. new Demo();

__set、__tostring等类的魔法方法的常量定义在源码级别是一起的,可是它并不是专属于某个类的魔法方法,它是所有的类共用的自动加载魔术方法,它将作为一个全局函数存在,那么Zend引擎是如何在类没有找到时调用这个方法的呢?

不管是使用new关键字创建类的实例,还是使用implement实现接口,或者继承某个类, 所有的这些操作都有可能调用__autoload函数,这几个操作在源码层都有一个共同点,它们在执行的时候都需要获取类的信息(接口在本质上也是一个类),它们在最终都会调用 zend_fetch_class(Zend/zend_execute_API.c)函数,这个函数本身没有多少内容,关键是它调用了zend_lookup_class_ex(Zend/zend_execute_API.c)函数,这个函数就是类的自动加载的真相所在.

在zend_lookup_class_ex函数中,我们看到程序会首先查询类符号表,如果存在类直接返回,如果不存在,就会执行我们所说的自动加载了,这里针对__autoload函数和spl相关的函数都做了处理,并且以第一参数和第二参数传递给Zend引擎的函数调用函数zend_call_function.

在zend_call_function函数中,它会判断第二参数是否存在函数,如果存在函数则只会调用第二个参数传递的函数(这里指SPL注册的函数),如果第二个函数没有值,则执行第一个参数传递过来的函数(这里指用户定义的__autoload函数),到这里,我想前面提到的两个方法同时存在的情况应该就有答案了,这也算是一篇基于的php教程了。