本文实例分析了Yii框架中使用PHPExcel的方法。分享给大家供大家参考,具体如下:PHPExcel是
本文实例分析了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::$enableIncludePath
为false,第三方类库就有了执行自己的autoload方法的机会,然后使用下面两行代码就能加载PHPExcel的类了:
Yii::$enableIncludePath = false;
Yii::import('application.vendors.phpexcel.PHPExcel', 1);
import时采用了force include的方式,这是因为PHPExcel.php在被require时才会注册autoloader,如果等到new PHPExcel时才注册,其他的类例如PHPExcel_IOFactory如果在这之前使用了,就会出现找不到类的错误。
个人认为我的这种办法是比较方便且优雅的,对比网上的其他办法好很多,下面列举的办法都或多或少有点问题,例如:
1、https://www.jb51.net/article/166128.htm,这种办法先将Yii自己的autoloader unregister了,会造成yii自己的类加载不上
2、https://www.jb51.net/article/166132.htm,这种办法还修改了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方法有机会执行。