Yii2使用PHPExcel读取excel

我的使用过程当中保存一些使用PHPExcel的经验,以便后来翻阅:
与PHP的Yii框架结合,能够轻松使用。并且根本不用网上所说的修改Yii的自动加载文件等方法。
具体使用方法:
下载phpoffice http://phpexcel.codeplex.com/releases/view/119187 (若是用composer的话,不须要手动下载)
下载后首级目录结构是 Classes, Documentation, Examples, changelog.txt, install.txt, license.txt.php

下载后要作的就是让Yii加载PHPExcel. lele模仿yii2-swiftmailer的加载方式。css

1. 修改项目根路径下的composer.json, 在"require"中加入"phpoffice/phpexcel": "dev-develop"json

2. 而后用命令行进入根目录,执行composer update(不用手动下载phpexcel,composer会自动下好。。。)swift

3. 修改vendor/composer/autoload_namespaces.php. 在return的Array中加上'PHPExcel' => array($vendorDir . '/phpoffice/phpexcel/Classes')数组

4. 在php中使用$PHPExcel = new \PHPExcel();来获得一个实例。切记,在Yii框架中,new时必定要加\, 否则会报类找不到的错误(lele不知道为何要加\, 有谁知道请在评论里说一下)。浏览器

OK, 能用了以后呢,写一些使用的代码:
$filePath = "../file/test.xlsx"; // 要读取的文件的路径
$PHPExcel = new \PHPExcel(); // 拿到实例,待会儿用
$PHPReader = new \PHPExcel_Reader_Excel2007(); // Reader很关键,用来读excel文件
if (!$PHPReader->canRead($filePath)) { // 这里是用Reader尝试去读文件,07不行用05,05不行就报错。注意,这里的return是Yii框架的方式。
    $PHPReader = new \PHPExcel_Reader_Excel5();
    if (!$PHPReader->canRead($filePath)) {
    $errorMessage = "Can not read file.";
    return $this->render('error', ['errorMessage' => $errorMessage]);
    }
}
$PHPExcel = $PHPReader->load($filePath); // Reader读出来后,加载给Excel实例
-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-
以上就能够读excel了,下面遍历输出这个二维表格。PHPExcel的方法名字比较语义化。
$allSheet = $PHPExcel->getSheetCount(); // sheet数
$currentSheet = $PHPExcel->getSheet(0); // 拿到第一个sheet(工做簿?)    
$allColumn = $currentSheet->getHighestColumn(); // 最高的列,好比AU. 列从A开始
$allRow = $currentSheet->getHighestRow(); // 最大的行,好比12980. 行从0开始
$result = new ReadFileResult(); // result是我本身写的一个存放结果的实体类
for ($currentRow = 1; $currentRow <= $allRow; $currentRow++) {
    echo $currentRow;
    $lineVal = [];
    for ($currentColumn="A"; $currentColumn <= $allColumn; $currentColumn++) {
    $val = $currentSheet->getCellByColumnAndRow(ord($currentColumn) - 65, $currentRow)->getValue(); // ord把字母转为ascii码,A->65, B->66....这儿的坑在于AU->65, 后面的U没有计算进去,因此用索引方式遍历是有缺陷的。
    array_push($lineVal, $val);
    }
    array_push($result->content, $lineVal);
}
看开发文档发现,还有两个很是好用的遍历方式。
一,用toArray把这个sheet转为二维数组。
$currentSheet->getStyle('A2:A6')->getNumberFormat()->setFormatCode('yyyy-mm-dd');  // 转为A2到A6的时间格式
$result->content = $currentSheet->toArray('', true, true); // 把当前sheet转为二维数组
二,用PHPExcel自带的行列迭代器。这样会稳妥许多。
foreach ($currentSheet->getRowIterator() as $row) { // 行迭代器
    $cellIterator = $row->getCellIterator(); // 拿到行中的cell迭代器
    $cellIterator->setIterateOnlyExistingCells(false); // 设置cell迭代器,遍历全部cell,哪怕cell没有值
    $lineVal = [];
    foreach ($cellIterator as $cell) {
        if ($cell->getDataType() == \PHPExcel_Cell_DataType::TYPE_NUMERIC) { // 这里是比较cell中数据类型是否是number
            $cellStyleFormat = $cell->getStyle( $cell->getCoordinate() )->getNumberFormat(); // 接下来这两句是拿到这个number的格式
            $formatCode = $cellStyleFormat->getFormatCode(); // 若是是普通的数字,formatCode将为General, 若是是如6/12/91 12:00的时间格式,formatCode将为/d/yy h:mm(反正就是时间格式了)
            echo $cell->getCoordinate() . " " . $formatCode; echo "<br>";
            if (preg_match("/m\/d\/yy h:mm/i", $formatCode)) {
                $value = gmdate("Y-m-d H:i:s", \PHPExcel_Shared_Date::ExcelToPHP($cell->getValue())); // 这里是将Excel的时间按格式转为PHP的时间
            } else {
                $value = $cell->getValue();
            }
        } else {
            $value = $cell->getValue();
        }
        if ($cell->getColumn() == 'I') { // 拿到列坐标
            $value = "0" . $value;
        }
        array_push($lineVal, $value . " " . $cell->getColumn());
    }
    array_push($result->content, $lineVal);
}
lele从yanhui_wei同窗一篇文章看到了TA对PHPExcel类及方法的总结,写的不错,特摘抄下来:
    
对于常见的excel报表操做,咱们须要掌握以下几个类库就能够了:


(1)PHPExcel:工做簿对象
excel文档处理对象主要用来管理咱们的excel文档,怎么来管理(经过属性和方法来管理)?你们知道,类主要是由属性和方法来组成,经过php程序的手段来管理excel文档,其实就是经过本对象的属性和方法来管理,下面咱们就来看一下PHPExcel类中都有那些属性和方法,这些属性和方法主要用来管理excel文档的那些方面
getProperties():得到当前活动状态工做表的属性对象,返回属性对象
getActiveSheet():得到当前活动状态的工做表,返回工做表对象
getActiveSheetIndex():得到当前活动状态工做表的索引值,返回int
setActiveSheetIndex():设置当前活动状态工做表的索引,返回工做表对象
getSheetByName():经过工做表名称获得当前工做表对象,返回工做表对象
getDefaultStyle():得到excel文档默认的样式(全部工做表的样式),返回样式对象
createSheet():在当前活动工做表后建立一个新的工做表
getSheetCount():得到excel文档中工做表的数量,返回int
getSheetNames():得到excel文档中全部工做表名称组成的数组


(2)PHPExcel_Worksheet:工做表对象
工做表对象,主要用来管理咱们的工做表,怎么管理?也是经过属性和方法来管理,可是工做表对象大部分状况下可经过excel文档对象来获取
toArray():把工做表中的数据转换成数组
fromArray():从数组中获取数据填充到工做表,返回工做表对象
getCell():得到单元格对象
getCellByColumnAndRow():经过列索引和行索引得到指定单元格,返回单元格对象
getDefaultStyle():得到工做表默认的样式,返回样式对象
getHighestColumn():得到工做表的最大列,返回列的名称
getColumnDimension():得到当前列
getStyle():得到指定单元格的样式,返回样式对象
getParent():得到父类对象,返回excel文档对象
getTitle():得到工做表的标题或名称,返回字符串类型
setCellValue():设置单元格的值,返回工做表对象或单元格对象,彻底取决于参数的值
setCellValueByColumnAndRow():经过列索引和行索引设置单元格的值,返回类型同上
setCellValueExplicit():设置单元格的值,并显示指定数据类型,返回工做表对象
setCellValueExplicitByColumnAndRow():经过列和行索引设置单元格值
setTitle():设置工做表标题


(3)PHPExcel_Cell:单元格对象
        
(4)PHPExcel_Style:样式对象,主要用来设置单元格的样式:对齐方式、字体、边框、填充等,跟咱们以前学过的css样式差很少,在这里若是想要设置对齐方式、字体大小、边框颜色、等等都是经过样式对象来完成的
getActiveCell():得到当前活动的单元格的名称,返回string;如,A1
getActiveSheet():得到当前活动的工做表,返回工做表对象
getAlignment():得到对齐方式对象,返回对齐方式对象
getBorders():得到边框对象,返回边框对象
getFill():得到填充对象
getFont():得到字体对象
setFont():设置字体,返回样式对象


(5)PHPExcel_Style_Alignment:对齐方式对象
getHorizontal():得到水平居中方式
getVertical():得到垂直居中方式
setHorizontal():设置水平居中方式,返回对齐方式对象
setVertical():设置垂直居中方式,返回对齐方式对象
居中方式:
HORIZONTAL_CENTER 
HORIZONTAL_CENTER_CONTINUOUS
HORIZONTAL_GENERAL 
HORIZONTAL_JUSTIFY 
HORIZONTAL_LEFT 
HORIZONTAL_RIGHT 
VERTICAL_BOTTOM 
VERTICAL_CENTER 
VERTICAL_JUSTIFY 
VERTICAL_TOP


(6)PHPExcel_Style_Font:字体对象
setBold():设置字体加粗
setColor():设置字体颜色
setItalic():设置字体倾斜
setName():设置字体名
setSize():设置字体大小
setUnderline():设置字体下划线


(7)PHPExcel_Writer_Excel5:写操做对象,主要用来输出xls文件
save(工做簿文件名):将工做簿对象中的数据保存到一个工做簿文件中


(8)PHPExcel_Writer_Excel2007:写操做对象,主要用于输出xlsx文件
save(工做簿文件名):将工做簿对象中的数据保存到一个工做簿文件中


(9)PHPExcel_Reader_Excel5:读操做对象,主要用于输入xls文件
canRead():当前reader对象是否可以读工做簿文件
load():从一个工做簿文件中加载工做簿对象,也就是将工做簿文件中的数据加载到工做簿对象中来管理


(10)PHPExcel_IOFactory:读写操做对象
createReader():根据参数的不一样,建立不一样的读对象:主要做用是读取工做簿文件中的数据
createWriter():根据参数的不一样,返回不一样的写对象:主要做用是将PHPExcel工做簿对象中的数据写入到一个工做簿文件中
load():从工做簿文件中加载PHPExcel工做簿对象,即:将工做簿文件中数据加载到PHPExcel工做簿对象中来管理


PHPExcel对象:是一个工做簿对象


include_once "PHPExcel/Writer/Excel5.php";//主要用于其它低版本,且文件名后缀为xls的文件,若是咱们但愿生成后缀名为xls格式的excel文件,建议引入此类库
include_once "PHPExcel/Writer/Excel2007.php";//主要用于excel2007格式,文件名后缀为xlsx的excel文件,若是咱们但愿生成后缀名为xlsx格式的excel文件,建议引入此类库
$objWriter = new PHPExcel_Writer_Excel5($objExcel);//建立一个文件格式写入对象实例,此对象主要用来写入内容到指定格式的文件,如,写入内容到后缀名为xls格式的excel文件等,用于其它板式的格式
$objWriter = new PHPExcel_Writer_Excel2007($objExcel);//建立一个文件格式写入对象实例,此对象主要用来写入内容到指定格式的文件,如,写入内容到后缀名为xls格式的excel文件等,用于excel2007格式
$objWriter->setOffice2003Compatibility(true);//兼容office2003


//设置文档基本属性  
$objProps = $objExcel->getProperties();  //获得PHPExcel_document文档对象
$objProps->setCreator("Zeal Li");       //设置做者
$objProps->setLastModifiedBy("Zeal Li"); //设置最后修改时间
$objProps->setTitle("Office XLS Test Document"); //设置标题
$objProps->setSubject("Office XLS Test Document, Demo");//设置主题
$objProps->setDescription("Test document, generated by PHPExcel.");//描 
$objProps->setKeywords("office excel PHPExcel"); //关键字
$objProps->setCategory("Test"); //分类
$objExcel->setActiveSheetIndex(0);//设置用户打开excel文件时,看到的首张sheet,若是没有设置,默认为最后一次操做的sheet
$objActSheet->setTitle('测试Sheet');//设置当前活动的工做簿名称


//根据单元格名称设置单元格内容,由PHPExcel根据传入的内容自动判断单元格的内容类型 
$objActSheet->setCellValue('A1', '字符串内容');  // 字符串内容 
$objActSheet->setCellValue('A2', 26);            // 数值 
$objActSheet->setCellValue('A3', true);          // 布尔值 
$objActSheet->setCellValue('A4', '=SUM(A2:A2)'); // 公式 


/显式指定单元格的内容类型为字符串类型  
$objActSheet->setCellValueExplicit('A5','847475847857487584',PHPExcel_Cell_DataType::TYPE_STRING);


//合并单元格  
$objActSheet->mergeCells('B1:C22'); 


//设置列的宽度  
$objActSheet->getColumnDimension('B')->setAutoSize(true);
$objActSheet->getColumnDimension('A')->setWidth(30);


//设置行的高度
$objPHPExcel->getActiveSheet()->getRowDimension('2')->setRowHeight(11.5);


//格式:主要用来对单元格进行操做,如,设置字体、设置对齐方式、设置边框等
$objStyleA5 = $objActSheet->getStyle('A5');//获取A5单元格的样式


//设置单元格的字体
$objFontA5 = $objStyleA5->getFont(); //得到字体
$objFontA5->setName('宋体');//设置字体名称 
$objFontA5->setSize(10);  //设置字体大小
$objFontA5->setBold(true);//设置字体加粗
$objFontA5->getColor()->setARGB('FF999999');//设置字体颜色


//设置单元格的对齐方式  
$objAlignA5 = $objStyleA5->getAlignment();//得到对齐方式
$objAlignA5->setHorizontal(PHPExcel_Style_Alignment::HORIZONTAL_RIGHT);//水平居右 
$objAlignA5->setVertical(PHPExcel_Style_Alignment::VERTICAL_CENTER);//垂直居中


//设置单元格的边框  
$objBorderA5 = $objStyleA5->getBorders();//获取边框 
$objBorderA5->getTop()->setBorderStyle(PHPExcel_Style_Border::BORDER_THIN);//边框样式
$objBorderA5->getTop()->getColor()->setARGB('FFFF0000');//顶部边框的颜色 
$objBorderA5->getBottom()->setBorderStyle(PHPExcel_Style_Border::BORDER_THIN); 
$objBorderA5->getLeft()->setBorderStyle(PHPExcel_Style_Border::BORDER_THIN);//左样式
$objBorderA5->getRight()->setBorderStyle(PHPExcel_Style_Border::BORDER_THIN);//右样式


//设置单元格的填充色
$objFillA5 = $objStyleA5->getFill();//填充 
$objFillA5->setFillType(PHPExcel_Style_Fill::FILL_SOLID);//填充类型 
$objFillA5->getStartColor()->setARGB('FFEEEEEE');


//计算单元格的值
$objPHPExcel->getActiveSheet()->setCellValue('B7', '=SUM(B5:C5)');
$objPHPExcel->getActiveSheet()->getCell('B7')->getCalculatedValue();


//若是要获取单元格的一个值,首先须要经过工做表的getCell方法获取到一个单元格对象,而后再经过单元格对象的getValue方法获得单元格的值,若是单元格的值是经过计算获得的,则须要使用getCalculatedValue方法获取单元格的值,设置单元格的值,咱们只须要经过工做表的setCellValue方法来设置便可


//$dateTimeNow=time();
$objPHPExcel->getActiveSheet()->setCellValue('C10', PHPExcel_Shared_Date::PHPToExcel( $dateTimeNow ));//41105.75
$objPHPExcel->getActiveSheet()->getStyle('C10')->getNumberFormat()->setFormatCode(PHPExcel_Style_NumberFormat::FORMAT_DATE_TIME4);//18:00:54,只是换了一种显示方式,并不会改变原来值的类型
echo gettype($objPHPExcel->getActiveSheet()->getCell('C10')->getValue());//double
echo $objPHPExcel->getActiveSheet()->getCell('C10')->getValue();//41105.75


//'2010-10-21'必定要放在引号中,不然显示的值为,1979               文本(推荐)
$objPHPExcel->setActiveSheetIndex(0)->setCellValueExplicit("D1", '2010-10-21', PHPExcel_Cell_DataType::TYPE_STRING); //特征:字符串类型都是居左显示


//添加一个新的worksheet  
$objExcel->createSheet();//建立一个新的工做表
$objExcel->getSheet(1)->setTitle('测试2');//设置当前工做表的标题


//保护单元格  
$objExcel->getSheet(1)->getProtection()->setSheet(true); 
$objExcel->getSheet(1)->protectCells('A1:C22', 'PHPExcel');  


//输出内容到excel文件,并将文件保存在服务器上
$objWriter->save("test.xls");


//强制输出内容到浏览器下载  
header("Content-Type: application/force-download"); 
header("Content-Type: application/octet-stream"); 
header("Content-Type: application/download"); 
header('Content-Disposition:inline;filename="'.$outputFileName.'"'); 
header("Content-Transfer-Encoding: binary"); 
header("Expires: Mon, 26 Jul 1997 05:00:00 GMT"); 
header("Last-Modified: " . gmdate("D, d M Y H:i:s") . " GMT"); 
header("Cache-Control: must-revalidate, post-check=0, pre-check=0"); 
header("Pragma: no-cache");  
$objWriter->save('php://output');//参数-表示直接输出到浏览器,供客户端下载


//excel读取对象
$PHPReader = new PHPExcel_Reader_Excel5();//建立一个excel文件的读取对象
$PHPExcel = $PHPReader->load($filePath);//读取一张excel表,返回excel文件对象
$currentSheet = $PHPExcel->getSheet(0);//读取excel文件中的第一张工做表
$allColumn = $currentSheet->getHighestColumn();//取得当前工做表最大的列号,如,E
$allRow = $currentSheet->getHighestRow();//取得当前工做表一共有多少行


//设置工做簿默认的样式
$objPHPExcel->getDefaultStyle()->getFont()->setName('Arial');
$objPHPExcel->getDefaultStyle()->getFont()->setSize(8);  


//合并单元格
$objPHPExcel->getActiveSheet()->mergeCells('A18:E22');服务器

相关文章
相关标签/搜索