本文实例分析了yii框架中使用phpexcel的方法。分享给大家供大家参考,具体如下:

phpexcel是一个比较好用的php读取excel文件的类库,今天遇到了在yii中如何加载phpexcel类文件的问题,因为yii的autoload机制是安装类名去找文件,即文件名就是相应的类名,而phpexcel的类文件命名方式则是:dir_dir_classname.php,即文件名把文件的目录名都记录了,这种命名方式yii肯定识别不了。怎么办?

其实phpexcel也有自己的autoload方法(phpexcel_autoloader::load() ),通过查看源码发现它也是通过spl_autoload_register函数注册的(在phpexcel_autoloader::register() 中),而我们知道php的autoload机制是,所有用spl_autoload_register函数注册的方法,都会在autoload时被spl_autoload_call函数执行一遍,因此我们只需要让phpexcel的autoload方法顺利注册上就行了。

如果了解yii的autoload机制,不清楚的可以看 附录 yii的autoload机制 ,可以知道,只要设置yii::$enableincludepathfalse,第三方类库就有了执行自己的autoload方法的机会,然后使用下面两行代码就能加载phpexcel的类了:

yii::$enableincludepath = false;
yii::import('application.vendors.phpexcel.phpexcel', 1);

import时采用了force include的方式,这是因为phpexcel.php在被require时才会注册autoloader,如果等到new phpexcel时才注册,其他的类例如phpexcel_iofactory如果在这之前使用了,就会出现找不到类的错误。

个人认为我的这种办法是比较方便且优雅的,对比网上的其他办法好很多,下面列举的办法都或多或少有点问题,例如:

1、,这种办法先将yii自己的autoloader unregister了,会造成yii自己的类加载不上

2、,这种办法还修改了phpexcel的autoloader,代价很大。

附录: yii的autoload机制

yii框架宣称自己的类加载方式很高效,是真正的“用时加载”,那究竟特别在哪里?今天研究了一下源码,发现其实是在代码级加了一层“路径缓存”。

我们知道,要实现自己的autoload方法,需要采用spl_autoload_register()函数注册一个autoload方法,yii注册的这个方法是yiibase::autoload(),稍后再讲解这个方法的逻辑。另外,yii一般都用yii::import($pathalias, $forceinclude=false)来加载相应的类(这个方法直接调用了yiibase::import() ),这个方法配合yiibase::autoload()就能实现“用时加载”了。

先说import的大致逻辑:

1、检查self::$_imports数组是否存在相应的$pathalias,如果有说明已经加载过了,直接返回类名或者目录名;否则继续第2步;

2、根据路径别名获得实际的路径名,并根据路径别名最后一部分是否是“*”可以知道要加载的路径别名是否是一个文件,如果是文件,去第3步;否则去第4步;

3、如果是$forceinclude是true,则立即require这个文件,并在$_imports数组中增加一项$alias => $classname;否则在数组$classmap中缓存一项$classname => $realpath

4、对于路径,会在数组$_includepaths中缓存这个路径,并且在$_imports数组中增加一项$alias => $realpath

5、结束。

因为$forceinclude默认都为false,所以import不会立即加载相应的类,等到使用时才真正加载,这是yiibase::autoload的工作。

autoload的大致逻辑:

1、检查类名是否已缓存在$classmap或$_coreclasses数组中,如果是则直接require相应的文件路径,$_coreclasses是框架自有类的映射表;否则去第2步;

2、检测yiibase::$enableincludepath是否为false,如果是则去第3步,否则直接include($classname . '.php')

3、遍历$includepaths数组,将目录名拼接上类名,检查是否为合法的php文件,如果是则include,然后跳出循环

4、结束。

需要注意的是,文档指出:如果要与其他类库一起使用,必须将$enableincludepath置为false,以便在yii::autoload()失败时,其他类库的autoload方法有机会执行。