导出业务数据为表格(csv)文件的一个简单通用的方法

jerry thinkphp 2015年11月19日 收藏
一个通用的下载导出方法
  1. /*
  2. * 通用方法
  3. * 导出大数据为CSV 
  4. * 参数依次传入 查询对象,CSV文件列头(键是数据表中的列名,值是csv的列名),文件名.对数据二次处理的函数;
  5. */
  6. private function DownloadCSV( $selectObject, $head, $fileName = time(), $callBack = '' ){
  7.     
  8.  if ( !is_object( $selectObject ) || !is_array( $head ) ) {
  9.     exit('参数错误!');
  10.  }

  11.  set_time_limit(0);
  12.  //下载头.
  13.  header ('Content-Type: application/vnd.ms-excel;charset=gbk');
  14.  header ('Content-Disposition: attachment;filename="'.$fileName.'.csv"');
  15.  header ('Cache-Control: max-age=0');
  16.     
  17.  //输出流;
  18.  $file = 'php://output';
  19.  $fp = fopen ( $file, 'a' );

  20.  $changCode = function( $changArr ) {
  21.      // 破Excel2003中文只支持GBK编码;
  22.      foreach ( $changArr as $k => $v ) {
  23.          $changArr [$k] = iconv ( 'utf-8', 'gbk', $v );
  24.      }
  25.      //返回一个 索引数组;
  26.      return array_values( $changArr );
  27.  };

  28.  //写入头部;
  29.  fputcsv ( $fp, $changCode( $head ) );
  30.     
  31.  //写入数据;
  32.  $pageSize = 100//每次查询一百条;
  33.  $page  = 1;//起始页码;
  34.  $list = array();

  35.  //查库;
  36.  $cloneObj = clone $selectObject;//因为thinkphp内部执行完select方法后会清空对象属性,所以clone;
  37.     
  38.  while ( $list = $cloneObj ->limit( $pageSize*( ($page++)-1 ), $pageSize )->select()  ) {

  39.     $cloneObj = clone $selectObject;
  40.             
  41.     //对查询结果二次处理
  42.     is_callable( $callBack ) && $list = call_user_func( $callBack, $list );

  43.     foreach ( $list as $key => $value ) {
  44.             
  45.          $value = array_intersect_key( $value, $head );//返回需要写入CSV的数据;
  46.          $value = array_merge( $head, $value );//利用此函数返回需要的顺序;
  47.          $value = $changCode( $value );
  48.          fputcsv ( $fp, $value );//写入数据;
  49.          flush();
  50.             
  51.     }

  52.     ob_flush();
  53.                 
  54.  }

  55.  exit();
  56. }
  1. //比如有一个方法 demo 在展示数据表时 还要有导出功能.

  2. public function demo(){

  3. /*

  4. 一堆判断搜索,之类的代码.

  5. */

  6. $list = $table->field($x)->join($l)->where($m);

  7. //比如$var是表单发送过来的 开启导出的"开关" 
  8. if ( $var == 1 )

  9.     $headerArr = array('数据表字段'=>'导出时CSV的列名');

  10.     $callBack = function( $list ) {
  11.         //对数据的二次处理;
  12.         return $list;
  13.     };

  14.     $this->DownloadCSV( $list, $headerArr, 'XXX'.date('YmdHi',time()), $callBack );

  15. }

  16. $list = $list->select();

  17. /* 分配到模板等等 */
利用thinkphp的model做的通用下载,可以节约很多做下载导出数据功能时也要写一堆 where 搜索判断的问题.