YII2框架中excel表格导出的方法详解


Posted in PHP onJuly 21, 2017

前言

表格的导入导出是我们在日常开发中经常会遇到的一个功能,正巧在最近的项目中做到了关于表格输出的功能,并且之前用TP的时候也做过,所以想着趁着这次功能比较多样的机会整理一下,方便以后需要的时候,或者有需要的朋友们参考学习,下面话不多说了,来一起看看详细的介绍:

本文是基于YII2框架进行开发的,不同框架可能会需要更改

一.普通excel格式表格输出

先是最普通的导出.xls格式的表格。首先先看一下表格在网站的显示效果

YII2框架中excel表格导出的方法详解

这里可以看到整个表格一共是7列。下面来看代码的实现。

1.controller文件

//导出统计

public function actionStatistics(){
 //设置内存
 ini_set("memory_limit", "2048M");
 set_time_limit(0);

 //获取用户ID
 $id = Yii::$app->user->identity->getId();

 //去用户表获取用户信息
 $user = Employee::find()->where(['id'=>$id])->one();

 //获取传过来的信息(时间,公司ID之类的,根据需要查询资料生成表格)
 $params = Yii::$app->request->get();
 $objectPHPExcel = new \PHPExcel();

 //设置表格头的输出
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('A1', '代理公司');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('B1', '收入');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('C1', '成本');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('D1', '稿件数');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('E1', '毛利(收入-成本)');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('F1', '毛利率(毛利/收入)*100%');
 $objectPHPExcel->setActiveSheetIndex()->setCellValue('G1', 'ARPU值');

 //跳转到recharge这个model文件的statistics方法去处理数据
 $data = Recharge::statistics($params);

 //指定开始输出数据的行数
 $n = 2;
 foreach ($data as $v){
 $objectPHPExcel->getActiveSheet()->setCellValue('A'.($n) ,$v['company_name']);
 $objectPHPExcel->getActiveSheet()->setCellValue('B'.($n) ,$v['company_cost']);
 $objectPHPExcel->getActiveSheet()->setCellValue('C'.($n) ,$v['cost']);
 $objectPHPExcel->getActiveSheet()->setCellValue('D'.($n) ,$v['num']);
 $objectPHPExcel->getActiveSheet()->setCellValue('E'.($n) ,$v['gross_margin']);
 $objectPHPExcel->getActiveSheet()->setCellValue('F'.($n) ,$v['gross_profit_rate']);
 $objectPHPExcel->getActiveSheet()->setCellValue('G'.($n) ,$v['arpu']);
 $n = $n +1;
 }
 ob_end_clean();
 ob_start();
 header('Content-Type : application/vnd.ms-excel');

 //设置输出文件名及格式
 header('Content-Disposition:attachment;filename="代理公司统计'.date("YmdHis").'.xls"');

 //导出.xls格式的话使用Excel5,若是想导出.xlsx需要使用Excel2007
 $objWriter= \PHPExcel_IOFactory::createWriter($objectPHPExcel,'Excel5');
 $objWriter->save('php://output');
 ob_end_flush();

 //清空数据缓存
 unset($data);
}

2.model文件

<?php
 namespace app\models;//model层的命名空间
 //注意要引用yii的arrayhelper
 use yii\helpers\ArrayHelper;
 use Yii;
 class Recharge extends \yii\db\ActiveRecord
 {
 //excel一次导出条数
 const EXCEL_SIZE = 10000;
 
 //统计导出
 public static function statistics($params){

 //导出时间条件
 if(empty($params['min'])){
 $date_max = date("Y-m-d",strtotime("-1 day"));
 $date_min = date("Y-m-d",strtotime("-31 day"));
 }else{
 $date_min = $params['min'];
 $date_max = $params['max'];
 }
 $where = '';
 $where .= '(`issue_date` BETWEEN '.'\''.$date_min.'\''.' AND '.'\''.$date_max.'\')';

 //查找指定数据
 $sql = 'select
 article.company_id,
 article.cost,
 article.company_cost
 from article WHERE article.status=2 AND '.$where;
 $article = Article::findBySql($sql)->asArray()->all();
 $article = ArrayHelper::index($article,null,'company_id');
 $companys = [];

 foreach ($article as $key=>$v){
 if(empty($key)){
 continue;
 }else{
 $number = count($v);
 $company = Company::find()->where(['id'=>$key])->select('name')->one();
 $company_name = $company['name'];
 $cost = 0;
 $company_cost = 0;
 foreach ($v as $n){
 $cost += $n['cost'];
 $company_cost += $n['company_cost'];
 }
 if($company_cost == 0){
 $company_cost =1;
 }

 //这里注意,数据的存储顺序要和输出的表格里的顺序一样
 $companys[] = [
 //公司名
 'company_name' => $company_name,

 //收入
 'company_cost' => $company_cost,

 //成本
 'cost' => $cost,

 //稿件数
 'num' => $number,

 //毛利
 'gross_margin' => $company_cost-$cost,

 //毛利率
 'gross_profit_rate' => round(($company_cost-$cost)/$company_cost*100,2).'%',

 //ARPU值
 'arpu' => round($company_cost/$number,2),
 ];
 }
 }
 return $companys;
 }
}

最终导出的效果(单元格大小导出后调整过)可以看到和网页显示的基本一样。

YII2框架中excel表格导出的方法详解

二.大数据表格导出

这时老板说了,我们不能只看总和的数据,最好是把详细数据也给导出来。既然老板发话了,那就做吧。还是按照第一种的方法去做,结果提示我php崩溃了,再试一次发现提示写入字节超出。打开php的配置文件php.ini

memory_limit = 128M

发现默认内存已经给到128M,应该是足够的了。于是我打开数据库一看,嚯!

接近83万条的数据进行查询并导出,可不是会出问题嘛!怎么办呢,于是我Google了一下,发现对于大数据(2万条以上)的导出,最好是以.csv的形式。不说废话,直接上代码

1.controller文件

//导出清单

public function actionInventory(){
 ini_set("memory_limit", "2048M");
 set_time_limit(0);
 $id = Yii::$app->user->identity->getId();
 $user = Employee::find()->where(['id'=>$id])->one();
 $params = Yii::$app->request->get();
 
 //类似的,跳转到recharge这个model文件里的inventory方法去处理数据
 $data = Recharge::inventory($params);
 
 //设置导出的文件名
 $fileName = iconv('utf-8', 'gbk', '代理商统计清单'.date("Y-m-d"));
 
 //设置表头
 $headlist = array('代理商','文章ID','文章标题','媒体','统计时间范围','状态','创建时间','审核时间','发稿时间','退稿时间','财务状态','成本','销售额','是否是预收款媒体类型','订单类别');
 header('Content-Type: application/vnd.ms-excel');
 
 //指明导出的格式
 header('Content-Disposition: attachment;filename="'.$fileName.'.csv"');
 header('Cache-Control: max-age=0');
 
 //打开PHP文件句柄,php://output 表示直接输出到浏览器
 $fp = fopen('php://output', 'a');
 
 //输出Excel列名信息
 foreach ($headlist as $key => $value) {
 //CSV的Excel支持GBK编码,一定要转换,否则乱码
 $headlist[$key] = iconv('utf-8', 'gbk', $value);
 }
 
 //将数据通过fputcsv写到文件句柄
 fputcsv($fp, $headlist);
 
 //每隔$limit行,刷新一下输出buffer,不要太大,也不要太小
 $limit = 100000;
 
 //逐行取出数据,不浪费内存
 foreach ($data as $k => $v) {
 //刷新一下输出buffer,防止由于数据过多造成问题
 if ($k % $limit == 0 && $k!=0) {
 ob_flush();
 flush();
 }
 $row = $data[$k];
 foreach ($row as $key => $value) {
 $row[$key] = iconv('utf-8', 'gbk', $value);
 }
 fputcsv($fp, $row);
 }
}

2.model文件(因为这部分我要处理的过多,所以只选择了部分代码),在查询数据那部分,因为要查的数据较多,所以可以结合我之前写的关于Mysql大数据查询处理的文章看一下

//清单导出

public static function inventory($params){
 //统计时间范围
 if(!empty($params['min']) && !empty($params['max'])){
 $ti = strtotime($params['max'])+3600*24;
 $max = date('Y-m-d',$ti);
 $time = $params['min'].'-'.$params['max'];
 $date_min = $params['min'];
 $date_max = $max;
 }else{
 $date_max = date('Y-m-d');
 $date_min = date('Y-m-d',strtotime("-31 day"));
 $time = $date_min.'-'.$date_max;
 }
 //查询数据
 if($params['state'] == 1){
 $where = '';
 $where .= ' AND (`issue_date` BETWEEN '.'\''.$date_min.'\''.' AND '.'\''.$date_max.'\')';
 $map = 'select
  company.name,
  article.id,
  article.title,
  media.media_name,
  article.status,
  article.created,
  article.audit_at,
  article.issue_date,
  article.back_date,
  article.finance_status,
  article.cost,
  article.company_cost,
  media.is_advance
  from article
  LEFT JOIN custom_package ON custom_package.id = article.custom_package_id
  LEFT JOIN `order` ON custom_package.order_id = `order`.`id`
  LEFT JOIN company ON company.id = article.company_id
  LEFT JOIN media ON media.id = article.media_id
  where article.status=2 and `order`.package=0'.$where;
 //查找的第一部分数据,使用asArray方法可以使我们查找的结果直接形成数组的形式,没有其他多余的数据占空间(注意:我这里查找分三部分是因为我要查三种不同的数据)
 $list1 = Article::findBySql($map)->asArray()->all();
 $where2 = '';
 $where2 .= ' AND (`issue_date` BETWEEN '.'\''.$date_min.'\''.' AND '.'\''.$date_max.'\')';
 $where2 .= ' AND (`back_date` > \''.$date_max.'\')';
 $map2 = 'select
  company.name,
  article.id,
  article.title,
  media.media_name,
  article.status,
  article.created,
  article.audit_at,
  article.issue_date,
  article.back_date,
  article.finance_status,
  article.cost,
  article.company_cost,
  media.is_advance
  from article
  LEFT JOIN custom_package ON custom_package.id = article.custom_package_id
  LEFT JOIN `order` ON custom_package.order_id = `order`.`id`
  LEFT JOIN company ON company.id = article.company_id
  LEFT JOIN media ON media.id = article.media_id
  where article.status=3 and `order`.package=0 '.$where2;
 //查找的第二部分数据
 $list2 = Article::findBySql($map2)->asArray()->all();
 $where3 = '';
 $where3 .= ' AND (`issue_date` BETWEEN '.'\''.$date_min.'\''.' AND '.'\''.$date_max.'\')';
 $map3 = 'select
  company.name,
  article.id,
  article.title,
  media.media_name,
  article.status,
  article.created,
  article.audit_at,
  article.issue_date,
  article.back_date,
  article.finance_status,
  article.cost,
  article.company_cost,
  media.is_advance
  from article
  LEFT JOIN custom_package ON custom_package.id = article.custom_package_id
  LEFT JOIN `order` ON custom_package.order_id = `order`.`id`
  LEFT JOIN company ON company.id = article.company_id
  LEFT JOIN media ON media.id = article.media_id
  where article.status=5 '.$where3;
 //查找的第三部分数据
 $list3 = Article::findBySql($map3)->asArray()->all();
 $list4 = ArrayHelper::merge($list1,$list2);
 $list = ArrayHelper::merge($list4,$list3);
 }
 //把结果按照显示顺序存到返回的数组中
 if(!empty($list)){
 foreach ($list as $key => $value){
 //代理公司
 $inventory[$key]['company_name'] = $value['name'];
 //文章ID
 $inventory[$key]['id'] = $value['id'];
 //文章标题
 $inventory[$key]['title'] = $value['title'];
 //媒体
 $inventory[$key]['media'] = $value['media_name'];
 //统计时间
 $inventory[$key]['time'] = $time;
 //状态
 switch($value['status']){
 case 2:
  $inventory[$key]['status'] = '已发布';
  break;
 case 3:
  $inventory[$key]['status'] = '已退稿';
  break;
 case 5:
  $inventory[$key]['status'] = '异常稿件';
  break;
 }
 //创建时间
 $inventory[$key]['created'] = $value['created'];
 //审核时间
 $inventory[$key]['audit'] = $value['audit_at'];
 //发稿时间
 $inventory[$key]['issue_date'] = $value['issue_date'];
 //退稿时间
 $inventory[$key]['back_date'] = $value['back_date'];
 //财务状态
 switch($value['finance_status']){
 case 0:
  $inventory[$key]['finance_status'] = '未到结算期';
  break;
 case 1:
  $inventory[$key]['finance_status'] = '可结算';
  break;
 case 2:
  $inventory[$key]['finance_status'] = '资源审批中';
  break;
 case 3:
  $inventory[$key]['finance_status'] = '财务审批中';
  break;
 case 4:
  $inventory[$key]['finance_status'] = '已结款';
  break;
 case 5:
  $inventory[$key]['finance_status'] = '未通过';
  break;
 case 6:
  $inventory[$key]['finance_status'] = '财务已审批';
  break;
 }
 //成本
 $inventory[$key]['cost'] = $value['cost'];
 //销售额
 $inventory[$key]['company_cost'] = $value['company_cost'];
 //是否是预售
 switch($value['is_advance']){
 case 0:
  $inventory[$key]['is_advance'] = '否';
  break;
 case 1:
  $inventory[$key]['is_advance'] = '是';
  break;
 case 2:
  $inventory[$key]['is_advance'] = '合同';
  break;
 }
 //订单类别
 switch($params['state']){
 case 1:
  $inventory[$key]['order_type'] = '时间区间无退稿完成订单';
  break;
 case 2:
  $inventory[$key]['order_type'] = '时间区间发布前退稿订单';
  break;
 case 3:
  $inventory[$key]['order_type'] = '时间区间发布后时间区间退稿订单';
  break;
 case 4:
  $inventory[$key]['order_type'] = '时间区间之前发布时间区间内退稿订单';
  break;
 case 5:
  $inventory[$key]['order_type'] = '异常订单';
  break;
 }
 }
 }else{
 $inventory[0]['company_name'] = '无数据导出';
 }
 return $inventory;
}

3.导出结果

YII2框架中excel表格导出的方法详解

导出数量

YII2框架中excel表格导出的方法详解

导出的文件

YII2框架中excel表格导出的方法详解

基本上可以保证整个过程在2~4秒内处理完成

三.合并单元格

老板一看做的不错,说你顺便把充值统计的导出也做了把,想想我都是处理过这么多数据的人了,还不是分分钟搞定的事?来,上原型图

YII2框架中excel表格导出的方法详解

噗,一口老血,话都说了,搞吧。在做的时候我发现,这次的导出主要是要解决单元格合并的问题。经过查资料发现,PHP本身是实现不了单元格合并的,于是我打算通过phpexcel来实现

如果是使用PHPExcel的话,基本操作是这样的(合并A1到E1)

$objPHPExcel->getActiveSheet()->mergeCells('A1:E1');
// 表格填充内容
$objPHPExcel->getActiveSheet()->setCellValue('A1','The quick brown fox.');

结果

YII2框架中excel表格导出的方法详解

或者这样的(合并A1到E4)

$objPHPExcel->getActiveSheet()->mergeCells('A1:E4');
$objPHPExcel->getActiveSheet()->setCellValue('A1','The quick brown fox.');

结果

YII2框架中excel表格导出的方法详解

这样并不能满足我的要求,首先它是一个一个合并的,其次我要显示的充值金额下面的类型是会变化的,不可能固定写死,然后每次都更改。所以放弃了这种方法。

后来在小伙伴的帮助下尝试用html转存excel的方法

1.方法文件(因为我要每天定时执行,所以并没有写到controller层)

public function actionExcelRechargeStatistics(){

 //先定义一个excel文件
 $filename = date('【充值统计表】('.date('Y-m-d').'导出)').".xls";
 header("Content-Type: application/vnd.ms-execl");
 header("Content-Type: application/vnd.ms-excel; charset=utf-8");
 header("Content-Disposition: attachment; filename=$filename");
 header("Pragma: no-cache");
 header("Expires: 0");
 //时间条件
 if(empty($params['min'])){
 $time = date('Y-m-d',strtotime("+1 day"));
 $where = ' created < \' '.$time.'\'';
 }else{
 $time = $params['min']+3600*24;
 $time_end = $params['max']+3600*24;
 $where = ' created <= \' '.$time_end.'\' AND created >= \''.$time.'\' ';
 }
 //充值类型列表
 $recharge_type = Recharge::find()->asArray()->all();
 if(empty($recharge_type)){
 $rechargelist[0]= '';
 }else{
 $rechargelist = ArrayHelper::map($recharge_type,'id','recharge_name');
 }
 $rechargelist1 = $rechargelist;
 $count = count($rechargelist1);
 //使用html语句生成显示的格式
 $excel_content = '<meta http-equiv="content-type" content="application/ms-excel; charset=utf-8"/>';
 $excel_content .= '<table border="1" style="font-size:14px;">';
 $excel_content .= '<thead>
   <tr>
   <th rowspan="2">ID</th>
   <th rowspan="2">公司名称</th>
   <th colspan='.$count.'>充值金额</th>
   <th rowspan="2">充值大小</th>
   <th rowspan="2">实际消费</th>
   <th rowspan="2">当前余额</th>
   </tr>
   <tr>
  ';
 foreach ($rechargelist1 as $v => $t){
 $excel_content .= '<th colspan="1">'.$t.'</th>';
 }
 $excel_content .= '</tr>
  </thead>';
 //查找最新的固化数据
 $search = RechargeStatistics::find()->where($where)->asArray()->all();
 if(!empty($search)){
 foreach ($search as $key => $value){
 $search[$key]['recharge'] = unserialize($value['recharge']);
 }
 }
 //html语句填充数据
 if(empty($search)){
 }else{
 foreach ($search as $k) {
 $excel_content .= '<td>'.$k['company_id'].'</td>';
 $excel_content .= '<td>'.$k['company_name'].'</td>';
 foreach ($rechargelist1 as $v=>$t){
 $price = 0;
 foreach ($k['recharge'] as $q=>$w){
  if($w['recharge_id'] == $v){
  $price = $w['price'];
  break;
  }
 }
 $excel_content .= '<td>'.$price.'</td>';
 }
 $excel_content .= '<td>'.$k['total'].'</td>';
 $excel_content .= '<td>'.$k['consume'].'</td>';
 $excel_content .= '<td>'.($k['total']-$k['consume']).'</td></tr>';
 }
 }
 $excel_content .= '</table>';
 echo $excel_content;
 die;
}

2.结果

YII2框架中excel表格导出的方法详解

到这里基本就完成所有的任务了!

总结

以上就是这篇文章的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
PHP静态新闻列表自动生成代码
Jun 14 PHP
PHP pathinfo()获得文件的路径、名称等信息说明
Sep 13 PHP
PHP中读写文件实现代码
Oct 20 PHP
关于svn冲突的解决方法
Jun 21 PHP
PHP Switch 语句之学习笔记
Sep 21 PHP
php 生成自动创建文件夹并上传文件的示例代码
Mar 07 PHP
PHP处理大量表单字段的便捷方法
Feb 07 PHP
php不使用copy()函数复制文件的方法
Mar 13 PHP
php使用高斯算法实现图片的模糊处理功能示例
Nov 11 PHP
PHP 根据key 给二维数组分组
Dec 09 PHP
PHP生成二维码与识别二维码的方法详解【附源码下载】
Mar 07 PHP
Cookie跨域问题解决方案代码示例
Nov 24 PHP
实例讲解YII2中多表关联的使用方法
Jul 21 #PHP
PHP实现表单提交数据的验证处理功能【防SQL注入和XSS攻击等】
Jul 21 #PHP
php实现基于pdo的事务处理方法示例
Jul 21 #PHP
php基于自定义函数记录log日志方法
Jul 21 #PHP
解决form中action属性后面?传递参数 获取不到的问题
Jul 21 #PHP
PHP实现的redis主从数据库状态检测功能示例
Jul 20 #PHP
PHP实现的mysql主从数据库状态检测功能示例
Jul 20 #PHP
You might like
php select,radio和checkbox默认选择的实现方法
2010/05/15 PHP
php上传图片存入数据库示例分享
2014/03/11 PHP
关于URL最大长度限制的相关资料查证
2014/12/23 PHP
CentOS 安装 PHP5.5+Redis+XDebug+Nginx+MySQL全纪录
2015/03/25 PHP
Thinkphp无限级分类代码
2015/11/11 PHP
Yii2简单实现多语言配置的方法
2016/07/23 PHP
PHP 7.1新特性的汇总介绍
2016/12/16 PHP
php实现数组纵向转横向并过滤重复值的方法分析
2017/05/29 PHP
PHP实现网站应用微信登录功能详解
2019/04/11 PHP
JQuery入门——移除绑定事件unbind方法概述及应用
2013/02/05 Javascript
JS localStorage实现本地缓存的方法
2013/06/22 Javascript
javascript仿php的print_r函数输出json数据
2013/09/13 Javascript
js中事件的处理与浏览器对象示例介绍
2013/11/29 Javascript
你所不了解的javascript操作DOM的细节知识点(一)
2015/06/17 Javascript
jQuery鼠标悬浮链接弹出跟随图片实例代码
2016/01/08 Javascript
VSCode使用之Vue工程配置eslint
2019/04/30 Javascript
JavaScript中的类型检查
2020/02/03 Javascript
JavaScript实现栈结构Stack过程详解
2020/03/07 Javascript
基于Vue中的父子传值问题解决
2020/07/27 Javascript
Vue页面渲染中key的应用实例教程
2021/01/12 Vue.js
[04:16]完美世界DOTA2联赛PWL S2 集锦第一期
2020/11/23 DOTA
Pycharm修改python路径过程图解
2020/05/22 Python
Python基于network模块制作电影人物关系图
2020/06/19 Python
Python3爬虫mitmproxy的安装步骤
2020/07/29 Python
使用CSS3来代替JS实现交互
2017/08/10 HTML / CSS
澳大利亚领先的在线美容商城:Adore Beauty
2017/04/14 全球购物
俄罗斯护发和专业化妆品购物网站:Hihair
2019/09/28 全球购物
电子商务专业个人的自我评价
2013/12/19 职场文书
招聘与培训专员岗位职责
2014/01/30 职场文书
《他得的红圈圈最多》教学反思
2014/04/24 职场文书
售后客服工作职责
2014/06/16 职场文书
地球物理学专业推荐信
2014/09/08 职场文书
八项规定个人对照检查材料思想汇报
2014/09/25 职场文书
2014年保育员个人工作总结
2014/12/02 职场文书
驳回起诉裁定书
2015/05/19 职场文书
Python django中如何使用restful框架
2021/06/23 Python