当前位置:首页 > PHP教程 > php高级应用 > 列表

详解如何实现phpoffice的excel导入功能解耦

发布:smiling 来源: PHP粉丝网  添加日期:2023-08-19 16:27:59 浏览: 评论:0 

在业务中开发中,表格的导入导出功能很常见。但是这里主要是使用PhpOffice类库介绍实现导入表格数据的功能。

冲突:

大部分的导入功能,就是通过点击按钮上传一张表格,然后后台读取表格数据根据业务整理后直接插入到数据库,最后再返回给前端。但是如果表格数据庞大,业务逻辑复杂的时候,就会导致导入那一块很臃肿不好维护。

解决方法:

处理方式是把导入与业务数据插入分离,所以在二者之间添加一个队列就可以了。导入只负责将表格数据存入队列。业务部分可以是单独的系统,最后就是消费队列中的数据了。这样一来,不但提升了导入速度,而且还让导入与系统解耦,不会因为异常而影响到其他业务。

编码:

1.下载PhpOffice。

composer repuire phpoffice/phpspreadsheet

2.导入导出代码。

  1. <?php 
  2. namespace app\common\helper; 
  3. use PhpOffice\PhpSpreadsheet\Spreadsheet; 
  4. use PhpOffice\PhpSpreadsheet\Writer\Xlsx; 
  5. use PhpOffice\PhpSpreadsheet\IOFactory; 
  6. use PhpOffice\PhpSpreadsheet\Cell\Coordinate; 
  7. use think\Exception; 
  8. class Excel 
  9.     // 导出 
  10.     public function outPut($data$columns$table = '导出文件'
  11.     { 
  12.         $spreadsheet = new Spreadsheet(); 
  13.         $sheet = $spreadsheet->getActiveSheet(); 
  14.         // 设置第一栏的标题 
  15.         foreach ($columns as $k => $v) { 
  16.             $sheet->setCellValue($k . "1"$v['title']); 
  17.         } 
  18.         //第二行起 设置内容 
  19.         $baseRow = 2; //数据从N-1行开始往下输出 这里是避免头信息被覆盖 
  20.         foreach ($data as $key => $value) { 
  21.             foreach ($columns as $k1 => $v1) { 
  22.                 $i = $key + $baseRow
  23.                 $sheet->setCellValue($k1 . $i$value[$v1['field']]); 
  24.             } 
  25.         } 
  26.         $writer = new Xlsx($spreadsheet); 
  27.         $filename = $table . date("Y-m-d", time()) . '_' . time() . '.xlsx'
  28.         $writer->save('./excel/' . $filename); 
  29.         return '/excel/' . $filename
  30.     } 
  31.     // 导入 
  32.     public function importExcel($file = ''$sheet = 0, $columnCnt = 0, &$options = []) 
  33.     { 
  34.         try { 
  35.             $file = iconv("utf-8""gb2312"$file); 
  36.             if (emptyempty($file) OR !file_exists($file)) { 
  37.                 throw new \Exception('文件不存在!'); 
  38.             } 
  39.             $objRead = IOFactory::createReader('Xlsx'); 
  40.             if (!$objRead->canRead($file)) { 
  41.                 $objRead = IOFactory::createReader('Xls'); 
  42.                 if (!$objRead->canRead($file)) { 
  43.                     throw new \Exception('只支持导入Excel文件!'); 
  44.                 } 
  45.             } 
  46.             /* 如果不需要获取特殊操作,则只读内容,可以大幅度提升读取Excel效率 */ 
  47.             emptyempty($options) && $objRead->setReadDataOnly(true); 
  48.             /* 建立excel对象 */ 
  49.             $obj = $objRead->load($file); 
  50.             /* 获取指定的sheet表 */ 
  51.             $currSheet = $obj->getSheet($sheet); 
  52.             //$currSheet = $obj->getSheetByName($sheet);      // 根据名字 
  53.             if (isset($options['mergeCells'])) { 
  54.                 /* 读取合并行列 */ 
  55.                 $options['mergeCells'] = $currSheet->getMergeCells(); 
  56.             } 
  57.             if (0 == $columnCnt) { 
  58.                 /* 取得最大的列号 */ 
  59.                 $columnH = $currSheet->getHighestColumn(); 
  60.                 /* 兼容原逻辑,循环时使用的是小于等于 */ 
  61.                 $columnCnt = Coordinate::columnIndexFromString($columnH); 
  62.             } 
  63.             /* 获取总行数 */ 
  64.             $rowCnt = $currSheet->getHighestRow(); 
  65.             $data = []; 
  66.             /* 读取内容 */ 
  67.             for ($_row = 1; $_row <= $rowCnt$_row++) { 
  68.                 $isNull = true; 
  69.                 for ($_column = 1; $_column <= $columnCnt$_column++) { 
  70.                     $cellName = Coordinate::stringFromColumnIndex($_column); 
  71.                     $cellId = $cellName . $_row
  72.                     $cell = $currSheet->getCell($cellId); 
  73.                     if (isset($options['format'])) { 
  74.                         /* 获取格式 */ 
  75.                         $format = $cell->getStyle()->getNumberFormat()->getFormatCode(); 
  76.                         /* 记录格式 */ 
  77.                         $options['format'][$_row][$cellName] = $format
  78.                     } 
  79.                     if (isset($options['formula'])) { 
  80.                         /* 获取公式,公式均为=号开头数据 */ 
  81.                         $formula = $currSheet->getCell($cellId)->getValue(); 
  82.                         if (0 === strpos($formula'=')) { 
  83.                             $options['formula'][$cellName . $_row] = $formula
  84.                         } 
  85.                     } 
  86.                     if (isset($format) && 'm/d/yyyy' == $format) { 
  87.                         /* 日期格式翻转处理 */ 
  88.                         $cell->getStyle()->getNumberFormat()->setFormatCode('yyyy/mm/dd'); 
  89.                     } 
  90.                     $data[$_row][$cellName] = trim($currSheet->getCell($cellId)->getFormattedValue()); 
  91.                     if (!emptyempty($data[$_row][$cellName])) { 
  92.                         $isNull = false; 
  93.                     } 
  94.                 } 
  95.                 if ($isNull) { 
  96.                     unset($data[$_row]); 
  97.                 } 
  98.             } 
  99.             return $data
  100.         } catch (\Exception $e) { 
  101.             throw $e
  102.         } 
  103.     } 

3.抽取指定的字段格式化Excel数据。

  1. return [ 
  2.     // 导入的表格标题 
  3.     "bidding" => [ 
  4.         "stock_no" => "编号"
  5.         "price" => "价格"
  6.         "mobile" => "手机"
  7.         "nickname" => "姓名" 
  8.     ] 
  9. ]; 
  10. // 格式化指定列数据(默认第一行表头) 
  11. public static function formattingCells(array $dataarray $cellConfig
  12.     $res = array_values($data); 
  13.     // 表头 
  14.     $header = $res[0]; 
  15.     $cellKeys = []; 
  16.     foreach ($header as $key => $value) { 
  17.         foreach ($cellConfig as $k => $v) { 
  18.             if ($value == $v) { 
  19.                 $cellKeys[$key] = $k
  20.             } 
  21.         } 
  22.     } 
  23.     if (count($cellKeys) != count($cellConfig)) { 
  24.         throw new Exception('表格不完整'); 
  25.     } 
  26.     // 需要添加过滤 
  27.     $temp = []; 
  28.     for ($i = 1; $i <= count($res) - 1; $i++) { 
  29.         foreach ($cellKeys as $m => $n) { 
  30.             $temp[$i][$n] = $res[$i][$m]; 
  31.         } 
  32.     } 
  33.     return array_values($temp); 

4.导入部分,上传接口。

  1. // 导入表格,上传接口 
  2. public function importExcel() 
  3.     $upload_file = $_FILES['files']['tmp_name']; 
  4.     $input = $this->input; 
  5.     // ID 
  6.     $id = isset($input['id']) ? $input['id'] : 0; 
  7.     // 默认取第一工作表 
  8.     $excelData = (new Excel())->importExcel($upload_file, 0); 
  9.     // 取Excel字段 
  10.     $config = config('excel_export.bidding'); 
  11.     $price_offer = Excel::formattingCells($excelData$config); 
  12.     // 判断每条记录的手机和价格格式 
  13.     // …… 
  14.     $jsonList = json_encode(compact('id''price_offer')); 
  15.     //$jsonList = json_encode($price_offer); 
  16.     // 入MQ 
  17.     $host = config("mq.host"); 
  18.     $options = config("mq.price_offer_import"); 
  19.     try { 
  20.         $mq = new ProductMQ($host$options); 
  21.         $mq->publish($jsonList); 
  22.         $mq->close(); 
  23.     } catch (\Exception $e) { 
  24.         return $this->jsonData(200, $e->getMessage()); 
  25.     } 
  26.     // 入MQ 
  27.     return $this->jsonData(200, '导入成功'); 

5.消费业务逻辑。

详解如何实现phpoffice的excel导入功能解耦

Tags: phpoffice excel导入

分享到: