这里有新鲜出炉的YII 2.0教程,程序狗速度看过来!
Yii Framework是一个基于组件、用于开发大型 Web 应用的高性能 PHP 框架。Yii提供了今日Web 2.0应用开发所需要的几乎一切功能。Yii是最有效率的PHP框架之一。Yii是创始人薛强的心血结晶,于2008年1月1日开始开发。
最近在研究PHP的Yii框架,很喜欢,碰到导出Excel的问题,研究了一下,就有了下面这篇文章,这篇文章主要给大家介绍了关于YII2框架中excel表格导出的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧。
前言
表格的导入导出是我们在日常开发中经常会遇到的一个功能,正巧在最近的项目中做到了关于表格输出的功能,并且之前用TP的时候也做过,所以想着趁着这次功能比较多样的机会整理一下,方便以后需要的时候,或者有需要的朋友们参考学习,下面话不多说了,来一起看看详细的介绍:
本文是基于YII2框架进行开发的,不同框架可能会需要更改
一.普通excel格式表格输出
先是最普通的导出.xls格式的表格。首先先看一下表格在网站的显示效果
这里可以看到整个表格一共是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;
- }
- }
最终导出的效果(单元格大小导出后调整过)可以看到和网页显示的基本一样。
二.大数据表格导出
这时老板说了,我们不能只看总和的数据,最好是把详细数据也给导出来。既然老板发话了,那就做吧。还是按照第一种的方法去做,结果提示我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.导出结果
导出数量
导出的文件
基本上可以保证整个过程在2~4秒内处理完成
三.合并单元格
老板一看做的不错,说你顺便把充值统计的导出也做了把,想想我都是处理过这么多数据的人了,还不是分分钟搞定的事?来,上原型图
噗,一口老血,话都说了,搞吧。在做的时候我发现,这次的导出主要是要解决单元格合并的问题。经过查资料发现,PHP本身是实现不了单元格合并的,于是我打算通过phpexcel来实现
如果是使用PHPExcel的话,基本操作是这样的(合并A1到E1)
- $objPHPExcel - >getActiveSheet() - >mergeCells('A1:E1');
- // 表格填充内容
- $objPHPExcel - >getActiveSheet() - >setCellValue('A1', 'The quick brown fox.');
结果
或者这样的(合并A1到E4)
- $objPHPExcel - >getActiveSheet() - >mergeCells('A1:E4');
- $objPHPExcel - >getActiveSheet() - >setCellValue('A1', 'The quick brown fox.');
结果
这样并不能满足我的要求,首先它是一个一个合并的,其次我要显示的充值金额下面的类型是会变化的,不可能固定写死,然后每次都更改。所以放弃了这种方法。
后来在小伙伴的帮助下尝试用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.结果
到这里基本就完成所有的任务了!
总结
以上就是这篇文章的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对PHPERZ的支持。
来源: http://www.phperz.com/article/17/1028/350684.html