Yii框架中使用PHPExcel的方法分析


Posted in PHP onJuly 25, 2019

本文实例分析了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、https://3water.com/article/166128.htm,这种办法先将Yii自己的autoloader unregister了,会造成yii自己的类加载不上

2、https://3water.com/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方法有机会执行。

希望本文所述对大家基于Yii框架的PHP程序设计有所帮助。

PHP 相关文章推荐
基于mysql的论坛(5)
Oct 09 PHP
解决163/sohu/sina不能够收到PHP MAIL函数发出邮件的问题
Mar 13 PHP
zend api扩展的php对象的autoload工具
Apr 18 PHP
利用php获取服务器时间的实现代码
Jun 07 PHP
使用php 获取时间今天明天昨天时间戳的详解
Jun 20 PHP
ie与session丢失(新窗口cookie丢失)实测及解决方案
Jul 15 PHP
学习php设计模式 php实现建造者模式
Dec 07 PHP
浅谈PHP面向对象之访问者模式+组合模式
May 22 PHP
PHP7 echo和print语句实例用法
Feb 15 PHP
PHP设计模式之单例模式定义与用法分析
Mar 26 PHP
PHP 7.4 新语法之箭头函数实例详解
May 09 PHP
laravel使用数据库测试注意事项
Apr 10 PHP
PHP保留两位小数的几种方法
Jul 24 #PHP
Yii框架使用PHPExcel导出Excel文件的方法分析【改进版】
Jul 24 #PHP
Yii Framework框架使用PHPExcel组件的方法示例
Jul 24 #PHP
PHP+Apache实现二级域名之间共享cookie的方法
Jul 24 #PHP
PHP容器类的两种实现方式示例
Jul 24 #PHP
使用swoole 定时器变更超时未支付订单状态的解决方案
Jul 24 #PHP
thinkphp5.1框架容器与依赖注入实例分析
Jul 23 #PHP
You might like
PHP JS Ip地址及域名格式检测代码
2013/09/27 PHP
php目录遍历函数opendir用法实例
2014/11/20 PHP
php实现中文字符截取防乱码方法汇总
2015/04/29 PHP
在Mac OS上自行编译安装Apache服务器和PHP解释器
2015/12/24 PHP
PHP实现通过文本文件统计页面访问量功能示例
2019/02/13 PHP
用javascript父窗口控制只弹出一个子窗口
2007/04/10 Javascript
Js日期选择器并自动加入到输入框中示例代码
2013/08/02 Javascript
javascript判断两个IP地址是否在同一个网段的实现思路
2013/12/13 Javascript
connect中间件session、cookie的使用方法分享
2014/06/17 Javascript
Jquery中扩展方法extend使用技巧
2014/08/24 Javascript
JavaScript采用递归算法计算阶乘实例
2015/08/04 Javascript
JS实现的车标图片提示效果代码
2015/10/10 Javascript
复杂的javascript窗口分帧解析
2016/02/19 Javascript
如何提高数据访问速度
2016/12/26 Javascript
nodejs模块nodemailer基本使用-邮件发送示例(支持附件)
2017/03/28 NodeJs
vue.js父组件使用外部对象的方法示例
2017/04/25 Javascript
NodeJS实现微信公众号关注后自动回复功能
2017/05/31 NodeJs
JS设计模式之观察者模式实现实时改变页面中金额数的方法
2018/02/05 Javascript
Vue 项目代理设置的优化
2018/04/17 Javascript
浅谈Vue.js路由管理器 Vue Router
2018/08/16 Javascript
nodeJS与MySQL实现分页数据以及倒序数据
2020/06/05 NodeJs
在Python中操作列表之List.append()方法的使用
2015/05/20 Python
Python smallseg分词用法实例分析
2015/05/28 Python
Python 爬虫爬取指定博客的所有文章
2016/02/17 Python
pandas使用get_dummies进行one-hot编码的方法
2018/07/10 Python
Python3编码问题 Unicode utf-8 bytes互转方法
2018/10/26 Python
Python爬虫实战之12306抢票开源
2019/01/24 Python
python安装本地whl的实例步骤
2019/10/12 Python
django 实现简单的插入视频
2020/04/07 Python
Django bulk_create()、update()与数据库事务的效率对比分析
2020/05/15 Python
Django实现随机图形验证码的示例
2020/10/15 Python
python 使用tkinter+you-get实现视频下载器
2020/11/17 Python
css3 clip实现圆环进度条的示例代码
2018/02/07 HTML / CSS
经典优秀个人求职自荐信格式
2013/09/25 职场文书
周末问候语大全
2015/11/10 职场文书
SQL解决未能删除约束问题drop constraint
2022/05/30 SQL Server