在 CakePHP 中使用 SimpleExcel 遇到的一个问题小记

前言

以前在公司做的一个项目中遇到的一个问题,还有意思的。

问题

大家有没有在 CakePHP 中使用过第三方库?我现在在引入了一个名为 SimpleExcel 的第三方库时遇到了一些问题。我觉得问题可能跟我把这个库的库文件放在了不正确的目录下有关。

正常情况下,该库在解压后,会有一个名为 SimpleExcel 的目录,该目录下的文件结构如下:

[email protected]:~/Desktop/faisalman-simple-excel-php-8d9fabc/src/SimpleExcel$ ls
Exception  Parser  SimpleExcel.php  Writer

使用的话,只需要在相应的 PHP 文件中引入该库就可以了,如下所示:

// test.php
// 该文件和 SimpleExcel 目录在同一个目录,所以下面的 require_once 中的路径为 ./SimpleExcel/SimpleExcel.php
use SimpleExcel\SimpleExcel;
require_once(‘./SimpleExcel/SimpleExcel.php‘);

现在在 CakePHP 中,我是这样做的,将 SimpleExcel 目录放在了app/controllers 目录下,然后在相应的控制器中加入了如下代码:

use SimpleExcel\SimpleExcel;
require_once(‘SimpleExcel/SimpleExcel.php‘);

但在访问该控制器时,总是出现问题,错误信息如下:

Warning (2): require_once(/opt/xplico/xi/app/controllers/SimpleExcel/ontroller.php) [function.require-once]: failed to open stream: No such file or directory [APP/controllers/SimpleExcel/SimpleExcel.php, line 43]

到底我哪里做错了?

答案

首先,请看下这篇文章《PHP autoload机制详解》。因为之前遇到的问题的根本原因就在于 SimpleExcel 库在使用 autoload 机制时不够严谨。

现在,重新看下那个错误提示:

Warning (2): require_once(/opt/xplico/xi/app/controllers/SimpleExcel/ontroller.php) [function.require-once]: failed to open stream: No such file or directory [APP/controllers/SimpleExcel/SimpleExcel.php, line 43]

从提示中可以看到,SimpleExcel.php 这个文件已经执行了,也就是说,之前我们将 SimpleExcel 库放在了 app/controllers 目录下,并在控制器中加入的如下代码是没有问题的。

use SimpleExcel\SimpleExcel;
require_once(‘SimpleExcel/SimpleExcel.php‘);

那为什么 SimpleExcel.php 会出现问题呢?现在定位到该文件出错的地方,如下所示:

if (!class_exists(‘Composer\\Autoload\\ClassLoader‘, false)){
    // autoload all interfaces & classes
    spl_autoload_register(function($class_name){
        // TODO(H): 错误就出在这里的require_once语句上
        if($class_name != ‘SimpleExcel‘) require_once(dirname(__FILE__).DIRECTORY_SEPARATOR.str_replace(‘\\‘, DIRECTORY_SEPARATOR, substr($class_name, strlen(‘SimpleExcel\\‘))).‘.php‘);
    });
}

为什么这里会出错,结合上面的错误信息,这里 require_once 了一个不存在的文件?那这个不存在的文件时怎么来的呢?这就涉及到 autoload 机制。这个不存在的文件的路径是 SimpleExcel.php 文件自身所在目录加上当前控制器需要的类的名字构成的。作为一个控制器,它当然需要 Controller 父类,但这个类并不在 /opt/xplico/xi/app/controllers/SimpleExcel 目录下,当然就出错了。

其实,SimpleExcel 库的作者本意是利用 autoload 机制加载 SimpleExcel.php 自身所需要的一些类(位于Exception、Parser 和 Writer 目录下),但这里对要加载的类的过滤不够严谨,就出现了上面我们遇到问题。

修改方式,注释上面的代码,加入如下代码,解决问题。

$currentDir = dirname(__FILE__);

require_once($currentDir . ‘/Parser/IParser.php‘);
require_once($currentDir . ‘/Parser/BaseParser.php‘);
require_once($currentDir . ‘/Parser/CSVParser.php‘);
require_once($currentDir . ‘/Parser/HTMLParser.php‘);
require_once($currentDir . ‘/Parser/JSONParser.php‘);
require_once($currentDir . ‘/Parser/TSVParser.php‘);
require_once($currentDir . ‘/Parser/XLSXParser.php‘);
require_once($currentDir . ‘/Parser/XMLParser.php‘);

require_once($currentDir . ‘/Writer/IWriter.php‘);
require_once($currentDir . ‘/Writer/BaseWriter.php‘);
require_once($currentDir . ‘/Writer/CSVWriter.php‘);
require_once($currentDir . ‘/Writer/HTMLWriter.php‘);
require_once($currentDir . ‘/Writer/JSONWriter.php‘);
require_once($currentDir . ‘/Writer/TSVWriter.php‘);
require_once($currentDir . ‘/Writer/XLSXWriter.php‘);
require_once($currentDir . ‘/Writer/XMLWriter.php‘);

require_once($currentDir . ‘/Exception/SimpleExcelException.php‘);

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-12-28 16:32:24

在 CakePHP 中使用 SimpleExcel 遇到的一个问题小记的相关文章

在cakephp中使用 phpunit

这个问题折腾一点时间,所以吸取教训,先上一个流程图 1.安装pear1)在命令行中,进入php安装目录,运行: go-pear.bat (可能出现报错情况)2)下载go-phar.bat至php安装目录,运行: php go-pear.phar 3)双击刚产生的PEAR_ENV.reg ,导入注册表4)此时pear应该安装成功了,更新至最新版本: pear upgrade (这一步挺重要的,phpunit貌似是使用pear的1.9.x才能安装) 这部分参考了 http://blog.csdn.n

c语言中如何设计和编写一个应用系统?

C程序中,如何设计和编写一个应用系统? 一. C语言文件的操作 1. 文件操作的基本方法: C语言将计算机的输入输出设备都看作是文件.例如,键盘文件.屏幕文件等. 向屏幕输出一个信息,例如"Hello"是 #include.h> int main() { printf("Hello\\n"); } 从键盘接收一个字符串然后显示是 #include.h> int main() { char a[10]; scanf("%s",&

System.getProperty()引起的悲剧--您的主机中的软件中止了一个已建立的连接

我已无法形容此刻我的心情.. 本来是已经写好的netty5的demo程序,server和client之间创建tcp长连接的..然后随便传点数据的简单demo..然后今天试了一下tcp粘包的例子,用到了System.getProperty()方法,悲剧就从此产生了..改好了程序一运行,服务器端正常启动,然后client端怎么也起不来,一启动就自动退出,完善了一下client端的异常捕获,然后就能偶尔报出些异常信息: 您的主机中的软件中止了一个已建立的连接 恰巧今天装了一些新软件和插件,挨个全都卸载

IOS中使用.xib文件封装一个自定义View

1.新建一个继承UIView的自定义view,假设类名叫做 MyAppVew #import <UIKit/UIKit.h> @class MyApp; @interface MyAppView : UIView @property(strong,nonatomic) MyApp*app; +(instancetype)appView; @end MyAppView.h #import "MyAppView.h" #import "MyApp.h" @i

在windows 2008 R2中SQl Server 2008中代理启动失败的一个原因总结

启动SQL代理的时候报错如下: 关调用实时(JIT)调试而不是此对话框的详细信息,请参见此消息的结尾. ************** 异常文本 **************System.NullReferenceException: 未将对象引用设置到对象的实例. Server stack trace: 在 Microsoft.SqlServer.Management.UI.VSIntegration.ObjectExplorer.ObjectExplorer.FindObjectExplore

BCB 中 Application->CreateForm 和 New 的一个区别

Application->Create 和 NEW 的一个区别 最近写windows服务的时候,恰巧碰到一个问题.我建立了一个DataModal,然后在Datamodal的OnCreate 事件中创建了另外一个类A的对象a.. a会去访问DataModal的全局指针.相当于下面这个东西: 发现,a访问的DataModal的全局指针有如下2种情况, 如果使用Application->CreateForm 创建,则a可以访问到DataModal的全局指针.即,用下面的方式创建: ? 如果用new的

4.6 找出二叉树中指定节点的下一个节点(中序后继),假定每个节点有父指针。

5 / 2 6 / \ 1 4 7 / 3 class Node{ Node left; Node right; Node parent; int val; } /** 1.如果有右子树,则结果为右子树的最左节点. 2.如果没有右子树,则需要回到父节点,如果当前节点是父节点的左子树,则父节点就是结果,如果不是继续向上再找父节点. */ public TreeNode inorderSucc(TreeNode n){ if(n==null) return null; if(n.right!=nul

输入任意一个字符串,如:“abDEe23dJfd343dPOddfe4CdD5ccv!23rr”。 取出该字符串中所有的字母组成一个新的字符串。

#include <stdio.h>#include <string.h> /*输入任意一个字符串,如:“abDEe23dJfd343dPOddfe4CdD5ccv!23rr”. 取出该字符串中所有的字母组成一个新的字符串.*/ void main(){ char oldStr[100] = "abDEe23dJfd343dPOddfe4CdD5ccv!23rr"; char newStr[100]; int i = 0,j = 0; for(i = 0; i

Caused by: java.io.IOException: 您的主机中的软件中止了一个已建立的连接。

异常详情 2017-07-16 10:55:26,218 ERROR [500.jsp] - java.io.IOException: 你的主机中的软件中止了一个已建立的连接. org.apache.catalina.connector.ClientAbortException: java.io.IOException: 你的主机中的软件中止了一个已建立的连接. at org.apache.catalina.connector.OutputBuffer.realWriteBytes(Output