Browse Source

Update README.md

yzh 4 years ago
parent
commit
6f5ab11e79
6 changed files with 553 additions and 53 deletions
  1. 476 5
      README.md
  2. 9 9
      examples/export_demo.php
  3. 1 1
      examples/import_demo.php
  4. 10 0
      src/DataFormatException.php
  5. 10 0
      src/PathException.php
  6. 47 38
      src/Pxlswrite.php

+ 476 - 5
README.md

@@ -1,22 +1,493 @@
 # xlswriter+yield+websocket
 # xlswriter+yield+websocket
-这是一个基于xlswriter+yield高性能操作excel的库,集成了websocket用于excel操作时的进度推送。
+这是一个基于xlswriter+yield高性能操作excel的库,集成了websocket用于excel操作时的进度推送。  
+本类继承自xlswriter扩展, 重新封装了一些常用的方法,同时保持基类的用法。  
+xlswriter文档<https://xlswriter-docs.viest.me/>
 
 
-## Installing
+功能特性:
+* 高性能读写百万级别的excel数据
+* 支持动态合并单元格
+* 字段定义和过滤
+* excel处理进度条
+
+# Installing
 ```
 ```
 composer require yzh0325/xlswrite
 composer require yzh0325/xlswrite
 ```
 ```
-## 安装xlswriter扩展 
+# 安装xlswriter扩展 
 * php>=7.0
 * php>=7.0
 * windows环境下 php>=7.2 ,xlswriter版本号大于等于 1.3.4.1
 * windows环境下 php>=7.2 ,xlswriter版本号大于等于 1.3.4.1
 ```
 ```
 pecl install xlswriter
 pecl install xlswriter
 ```
 ```
-## WebSocket 
+# WebSocket 
 * 服务端采用swoole搭建,需要安装swoole扩展 swoole>=4.4.*,需在cli模式下运行
 * 服务端采用swoole搭建,需要安装swoole扩展 swoole>=4.4.*,需在cli模式下运行
 * 客户端采用textalk/websocket的client作为websocket客户端,可在php-fpm模式下运行
 * 客户端采用textalk/websocket的client作为websocket客户端,可在php-fpm模式下运行
 
 
-## examples
+# examples
 见 examples/ 
 见 examples/ 
 * xlswrite_demo.html 前端demo
 * xlswrite_demo.html 前端demo
 * export_demo.php excel导出demo
 * export_demo.php excel导出demo
 * import_demo.php excel导入demo
 * import_demo.php excel导入demo
+
+## 简单导出demo:
+```
+$fileObj = new Pxlswrite(['path' => __DIR__]);
+$filePath = $fileObj
+    ->fileName('123.xlsx','sheet1') //fileName 会自动创建一个工作表,你可以自定义该工作表名称,工作表名称为可选参数
+    ->field([
+        'username'=>['name'=>'用户名'],
+        'age'=>['name'=>'年龄']
+    ]) //设置字段&表格头
+    ->setDataByGenerator('generateData') //通过回调生成器方法向excel填充数据
+    ->output(); //输出excel文件到磁盘,返回文件路径
+$fileObj->download($filePath); //下载excel文件
+/**
+* 数据生成器方法 封装数据获取逻辑 通过yield返回 节省内存
+*/
+function generateData(){
+    for($i=0;$i<10000;$i++){
+        yield [
+                [
+                    'username' => '匿名用户'.rand(1,9999),
+                    'age' => rand(1,100),  
+                 ]
+              ];
+    }
+}
+```
+
+## 使用WebSocket推送excel处理进度
+
+实现过程:前端与websocket服务器先建立连接,连接建立好之后服务端会发送一条消息 {'status' => 'onopen', 'fd' => $request->fd},
+前端将fd保存起来,在请求处理excel接口时将fd参数传给后台,后台在推送进度消息时将fd参数封装到消息体里在发送给websocket服务器
+,websocket服务器根据fd参数在推送消息给指定的前端,这就完成了处理excel数据的实时进度交互。
+
+* 前端首次连接时的消息体:['status' => 'onopen', 'fd' => 1]
+* 处理过程中的消息体:['status' => 'processing', 'process' => 100, 'fd'=>1]
+
+### 开启websocket 服务端 
+```
+php ./src/WebSocket/WebSocketServer.php
+```
+### web前端 
+```
+var layer = layui.layer, $ = layui.jquery, upload = layui.upload;
+var loading, fileLoading;
+var wsServer = 'ws://127.0.0.1:9502';
+var websocket = new WebSocket(wsServer);
+var fd = 0;//客户端socket id  
+websocket.onopen = function (evt) {
+     console.log("Connected to WebSocket server.");
+};
+websocket.onclose = function (evt) {
+    console.log("Disconnected");
+};
+websocket.onmessage = function (evt) {
+    var data = JSON.parse(evt.data);
+    if (data.status == 'onopen') {
+        fd = data.fd;
+    } else {
+        $('#process').text('已处理' + data.process + '条数据...');
+        // console.log(data.process);
+    }
+    // console.log('Retrieved data from server: ' + data);
+};
+websocket.onerror = function (evt, e) {
+    console.log('Error occured: ' + evt.data);
+};
+
+//导出excel
+    function exporData() {
+        loading = layer.load(2, {
+            shade: [0.1, '#fff'],
+            content: '正在到导出,请耐心等待...',
+            id: 'process',
+            success: function (layero) {
+                layero.find('.layui-layer-content').css({
+                    'padding-top': '40px',//图标与样式会重合,这样设置可以错开
+                    'width': '200px',//文字显示的宽度
+                    'text-indent': '-4rem',
+                });
+            }
+        });
+        //请求后台excel文件生成接口
+        $.get('/export_demo.php', {fd: fd}, function (res) {
+            if (res.code == 1) {
+                layer.close(loading);
+                layer.msg(res.msg, {time: '1000'}, function () {
+                    //请求后台下载地址
+                    window.location.href = res.url;
+                });
+            }
+        }, 'json');
+    }
+```
+### php 后端调用 WebSocketClient 推送消息
+```
+$fileObj = new Pxlswrite(['path' => __DIR__]);
+//实例化WebSocketClient--需要推送进度才实例化
+$pushHandle = new WebSocketClient('ws://127.0.0.1:9502', $_GET['fd']);
+$filePath = $fileObj->fileName('123.xlsx','sheet1')
+    ->field($field)//设置字段&表格头
+    ->setDataByGenerator('generateData', [], [], $pushHandle)//设置数据 回调生成器方法获取数据,$pushHandle 用于推送,可选参数
+    ->output();//输出excel文件到磁盘
+```
+## 样式设置
+支持的样式如下:
+```
+$style = [
+    'align' => [Pxlswrite::FORMAT_ALIGN_CENTER, Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER],//对齐 [x,y]
+    'border' => Pxlswrite::BORDER_SLANT_DASH_DOT,//单元格边框
+    'background' => Pxlswrite::COLOR_RED,//单元格背景色
+    'fontColor' => Pxlswrite::COLOR_BLUE,//字体颜色
+    'fontSize' => 30,//字体大小
+    'font' => 'FontName',//设置字体 字体名称,字体必须存在于本机
+    'number' => '#,##0',//数字格式化
+    'bold' => true,//粗题
+    'strikeout' => false,//文本删除线
+    'wrap' => true,//文本换行
+    'italic' => true,//斜体
+];
+```
+样式相关常量
+```
+    const FORMAT_ALIGN_LEFT = Format::FORMAT_ALIGN_LEFT;                                    // 水平左对齐
+    const FORMAT_ALIGN_CENTER = Format::FORMAT_ALIGN_CENTER;                                // 水平剧中对齐
+    const FORMAT_ALIGN_RIGHT = Format::FORMAT_ALIGN_RIGHT;                                  // 水平右对齐
+    const FORMAT_ALIGN_FILL = Format::FORMAT_ALIGN_FILL;                                    // 水平填充对齐
+    const FORMAT_ALIGN_JUSTIFY = Format::FORMAT_ALIGN_JUSTIFY;                              // 水平两端对齐
+    const FORMAT_ALIGN_CENTER_ACROSS = Format::FORMAT_ALIGN_CENTER_ACROSS;                  // 横向中心对齐
+    const FORMAT_ALIGN_DISTRIBUTED = Format::FORMAT_ALIGN_DISTRIBUTED;                      // 分散对齐
+    const FORMAT_ALIGN_VERTICAL_TOP = Format::FORMAT_ALIGN_VERTICAL_TOP;                    // 顶部垂直对齐
+    const FORMAT_ALIGN_VERTICAL_BOTTOM = Format::FORMAT_ALIGN_VERTICAL_BOTTOM;              // 底部垂直对齐
+    const FORMAT_ALIGN_VERTICAL_CENTER = Format::FORMAT_ALIGN_VERTICAL_CENTER;              // 垂直剧中对齐
+    const FORMAT_ALIGN_VERTICAL_JUSTIFY = Format::FORMAT_ALIGN_VERTICAL_JUSTIFY;            // 垂直两端对齐
+    const FORMAT_ALIGN_VERTICAL_DISTRIBUTED = Format::FORMAT_ALIGN_VERTICAL_DISTRIBUTED;    // 垂直分散对齐
+
+    const UNDERLINE_SINGLE = Format::UNDERLINE_SINGLE;                                      // 单下划线
+//    const UNDERLINE_DOUBLE = Format::UNDERLINE_DOUBLE;                                      // 双下划线
+    const UNDERLINE_SINGLE_ACCOUNTING = Format::UNDERLINE_SINGLE_ACCOUNTING;                // 会计用单下划线
+    const UNDERLINE_DOUBLE_ACCOUNTING = Format::UNDERLINE_DOUBLE_ACCOUNTING;                // 会计用双下划线
+
+    const BORDER_THIN = Format::BORDER_THIN;                                                // 薄边框风格
+    const BORDER_MEDIUM = Format::BORDER_MEDIUM;                                            // 中等边框风格
+    const BORDER_DASHED = Format::BORDER_DASHED;                                            // 虚线边框风格
+    const BORDER_DOTTED = Format::BORDER_DOTTED;                                            // 虚线边框样式
+    const BORDER_THICK = Format::BORDER_THICK;                                              // 厚边框风格
+    const BORDER_DOUBLE = Format::BORDER_DOUBLE;                                            // 双边风格
+    const BORDER_HAIR = Format::BORDER_HAIR;                                                // 头发边框样式
+    const BORDER_MEDIUM_DASHED = Format::BORDER_MEDIUM_DASHED;                              // 中等虚线边框样式
+    const BORDER_DASH_DOT = Format::BORDER_DASH_DOT;                                        // 短划线边框样式
+    const BORDER_MEDIUM_DASH_DOT = Format::BORDER_MEDIUM_DASH_DOT;                          // 中等点划线边框样式
+    const BORDER_DASH_DOT_DOT = Format::BORDER_DASH_DOT_DOT;                                // Dash-dot-dot边框样式
+    const BORDER_MEDIUM_DASH_DOT_DOT = Format::BORDER_MEDIUM_DASH_DOT_DOT;                  // 中等点划线边框样式
+    const BORDER_SLANT_DASH_DOT = Format::BORDER_SLANT_DASH_DOT;                            // 倾斜的点划线边框风格
+
+    const COLOR_BLACK = Format::COLOR_BLACK;
+    const COLOR_BLUE = Format::COLOR_BLUE;
+    const COLOR_BROWN = Format::COLOR_BROWN;
+    const COLOR_CYAN = Format::COLOR_CYAN;
+    const COLOR_GRAY = Format::COLOR_GRAY;
+    const COLOR_GREEN = Format::COLOR_GREEN;
+    const COLOR_LIME = Format::COLOR_LIME;
+    const COLOR_MAGENTA = Format::COLOR_MAGENTA;
+    const COLOR_NAVY = Format::COLOR_NAVY;
+    const COLOR_ORANGE = Format::COLOR_ORANGE;
+    const COLOR_PINK = Format::COLOR_PINK;
+    const COLOR_PURPLE = Format::COLOR_PURPLE;
+    const COLOR_RED = Format::COLOR_RED;
+    const COLOR_SILVER = Format::COLOR_SILVER;
+    const COLOR_WHITE = Format::COLOR_WHITE;
+    const COLOR_YELLOW = Format::COLOR_YELLOW;
+```
+样式设置的相关方法
+```
+/**
+ * 行单元格样式
+ * @param string $range  单元格范围
+ * @param double $height 单元格高度
+ * @param resource|array $formatHandler  单元格样式
+ * @return $this
+ * @throws \Exception
+ */
+setRow($range, $height, $formatHandler = null);
+/**
+ * 列单元格样式
+ * @param $range string 单元格范围
+ * @param $width double 单元格宽度
+ * @param null $formatHandler resource|array 单元格样式
+ * @return $this
+ * @throws \Exception
+ */
+setColumn($range, $width, $formatHandler = null)
+/**
+ * 全局默认样式
+ * @param resource|array $formatHandler style
+ * @return $this
+ * @throws DataFormatException
+ */
+defaultFormat($formatHandler)
+ /**
+ * 合并单元格
+ * @param string $scope   单元格范围
+ * @param string $data    data
+ * @param resource|array $formatHandler 合并单元格的样式
+ * @return $this
+ * @throws DataFormatException
+ */
+mergeCells($scope, $data, $formatHandler = null)
+```
+示例
+```
+$fileObj = new Pxlswrite(['path' => __DIR__]);
+$filePath = $fileObj->fileName('123.xlsx','sheet1')
+    ->field($field)//设置字段&表格头
+    ->setDataByGenerator('generateData')//设置数据
+    ->setRow('A1:A3', 80, ['bold'=>true]) //设置单元行样式 A1:A3 单元格范围 80行高 ['blod'=>true] 加粗
+    ->setColumn('A:F', 20, ['background' => Pxlswrite::COLOR_GRAY, 'align' => [Pxlswrite::FORMAT_ALIGN_CENTER, Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER]]) //设置单元列样式
+    ->output();//输出excel文件到磁盘
+```
+注意:设置行与行/列与列样式 交集范围会覆盖;行样式优先于列样式;全局默认样式不会被覆盖,谨慎使用
+
+## 动态合并单元格
+* 通用合并,根据数据的值比较 自动进行 行合并
+
+优点:数据层不需要怎么处理,将数据库查询出来的二维数组直接传入即可。  
+缺点:无法满足像订单一样的 存在多个订单时间值是一样的,合并就会存在问题。
+
+* 订单类型合并,将原始数据处理成指定的数据格式在传入此类自动进行 行合并
+
+优点:根据传入的数据格式进行合并,合并更加精准不会存在问题。  
+缺点:在数据层需要处理成指定的数据格式,数据格式如下:
+```
+$data = [
+    [
+        'order'=>'20200632555' ,
+        'time'=>'2020-06-30 12:30:10',
+        'username'=>'张三',
+        'address'=>'成都',
+        'phone'=>'17756891562',
+        'item'=> [
+            [
+                'itemnumer'=>'2020515',
+                'productname'=>'商品1',
+                'mark'=>'备注',
+            ],
+        ],
+    ]
+];
+```
+字段数量,名称没有限制,只支持一个item(也可以叫其他名字)二维数组,item数组里面的个数没有限制。
+### 通用合并demo
+函数原型
+```
+ /**
+ * 通过生成器逐行向表格插入数据,
+ * 设置过field才支持动态单元格合并,
+ * 可以根据指定的字段 通过值比较自动进行 行合并
+ * @param callable $_generator 回调数据生成器方法 返回的数据格式是二维数组 如下字段名数量不限
+ * [['id'=>1,'name'=>'张三','age'=>'18']]
+ * @param array $_mergeColumn 需要合并的字段
+ * @param array $_mergeColumnStyle 合并单元格的样式
+ * @param int $_index 单元格行偏移量 合并单元格的起始位置
+ * @param WebSocketClient|null $_pushHandle 用于推送的WebSocketClient
+ * @return Pxlswrite
+ * @throws DataFormatException
+ */
+setDataByGenerator($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1
+```
+示例
+```
+//定义字段
+$field = [
+    'id' => ['name' => 'title'],
+    'c1' => ['name' => 'age'],
+    'c2' => ['name' => 'year'],
+    'c3' => ['name' => 'kk'],
+];
+$fileObj = new Pxlswrite(['path' => __DIR__ ]);
+$filePath = $fileObj->fileName('123.xlsx');
+    ->field($field)//设置字段&表格头
+    ->setDataByGenerator('generateData', ['c1', 'c2'], ['align' => [Pxlswrite::FORMAT_ALIGN_CENTER, Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER]])//设置数据 并自动合并单元格
+    ->output();//输出excel文件到磁盘
+//数据生成器
+function generateData(){
+    for($i=0;$i<10000;$i++){
+        yield [
+                ['id'=>$i,'c1'=>$i+1,'c2'=>$i+2,'c3'=>$i+3];
+            ];
+    }
+}
+```
+
+### 订单类型合并demo
+函数原型
+```
+/**
+ * 设置订单数据 根据数据可以合并指定的字段,需要遵循以下数据格式
+ * @param callable $_generator 数据生成器方法 返回数据格式如下,字段数量名称不限,只支持一个item二维数组
+ * [
+ *    [
+ *        'order'=>'20200632555' ,
+ *        'time'=>'2020-06-30 12:30:10',
+ *        'username'=>'张三',
+ *        'address'=>'成都',
+ *        'phone'=>'17756891562',
+ *        'item'=> [
+ *            [
+ *                'itemnumer'=>'2020515',
+ *                'productname'=>'商品1',
+ *                'mark'=>'备注',
+ *            ],
+ *        ],
+ *    ]
+ * ];
+ * @param array $_mergeColumn 需要合并的字段
+ * @param array $_mergeColumnStyle 合并单元格样式
+ * @param WebSocketClient|null $_pushHandle WebSocketClient对象 用于推送进度
+ * @param int $_index 单元格行偏移量 合并单元格的起始位置
+ * @return $this
+ * @throws DataFormatException
+ */
+setOrderData($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1)
+``` 
+示例
+```
+//定义字段
+$orderField =  [
+    'order'=>['name'=>'订单号'] ,
+    'time'=>['name'=>'下单时间'],
+    'username'=>['name'=>'用户名'],
+    'address'=>['name'=>'地址'],
+    'phone'=>['name'=>'手机号'],
+    'itemnumer'=>['name'=>'子订单号'],
+    'productname'=>['name'=>'商品名称'],
+    'amount'=>['name'=>'数量'],
+    'mark'=>['name'=>'备注'],
+];
+$fileObj = new Pxlswrite(['path' => __DIR__ ]);
+$filePath = $fileObj->fileName('123.xlsx');
+    ->field($orderField)//设置字段&表格头
+    ->setOrderData('generateOrderData',['order','time'],['align' => [Pxlswrite::FORMAT_ALIGN_CENTER, Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER]])//设置订单类型数据并自动合并单元格
+    ->output();//输出excel文件到磁盘
+//订单数据生成器 一个订单有多个item
+function generateOrderData(){
+    for ($i=0;$i<100;$i++){
+        $order = [];
+        $range = mt_rand(0,5);
+        for($k = 0;$k<$range;$k++){
+            $orderId = date('YmdHis').rand(1000,9999);
+            $range2 = mt_rand(1,5);
+            $item = [];
+            for($j = 0;$j<$range2;$j++){
+                $item[] = [
+                    'itemnumer'=>$orderId,
+                    'productname'=>'商品'.rand(10,99),
+                    'amount'=>'1',
+                    'mark'=>'备注',
+                ];
+            }
+            $order[] = [
+                'order'=>$orderId ,
+                'time'=>date('Y-m-d H:i:s'),
+                'username'=>'张三',
+                'address'=>'成都',
+                'phone'=>'17756891562',
+                'item'=> $item,
+            ];
+        }
+        yield $order;
+    }
+}
+```
+## 合并单元格
+函数原型
+```
+/**
+ * 合并单元格
+ * @param string $scope   单元格范围
+ * @param string $data    data
+ * @param resource|array $formatHandler  合并单元格的样式
+ * @return $this
+ * @throws DataFormatException
+ */
+mergeCells($scope, $data, $formatHandler = null)
+```
+示例
+```
+$fileObj->fileName("test.xlsx")
+  ->mergeCells('A1:C1', 'Merge cells')
+  ->output();
+```
+## 定义字段&格式化字段
+函数原型
+```
+/**
+ * 设置字段&表格头
+ * @param array $field 字段定义数组 数据格式如下
+ * [
+ *  'name' => ['name' => '姓名','callback'=>'functionName'],
+ *  'age' => ['name' => '年龄'],
+ * ]
+ * @return $this
+ * @throws DataFormatException
+ */
+function field($field)
+```
+示例
+```
+//定义字段
+$field = [
+    'name' => ['name' => '姓名'],
+    'year' => ['name' => '出生年份'],
+    'age' => ['name' => '年龄','callback'=>'ageFormat']//callback 回调方法处理格式化值 可以是函数/对象方法
+];
+$fileObj = new Pxlswrite(['path' => __DIR__ ]);
+$fileObj->fileName('1234.xlsx')
+    ->field($field)//设置字段&表格头
+    ->data([
+        ['Rent', 1999,0],
+        ['Gas',  1996,0],
+        ['Food', 1998,0],
+        ['Gym',  1995,0],
+    ])
+    ->output();//输出excel文件到磁盘
+//格式化字段值
+function ageFormat($v, $values)
+{
+    return date(Y) - $values['year'];
+}
+```
+## 游标读取excel 分段写入数据库
+函数原型
+```
+/**
+ * 游标读取excel,分段插入数据库
+ * @param callable $_func 方法名 回调数据插入的方法
+ * @param WebSocketClient|null $_pushHandle
+ * @param array $_dataType 可指定每个单元格数据类型进行读取
+ */
+importDataByCursor($_func, WebSocketClient $_pushHandle = null, array $_dataType = [])
+```
+示例
+```
+$fileObj = new Pxlswrite(['path' => dirname($_GET['file'])]);
+$fileInfo = explode('/', $_GET['file']);
+$fileName = end($fileInfo);
+//实例化WebSocketClient
+$pushHandle = new WebSocketClient('ws://127.0.0.1:9502',$_GET['fd']);
+//          打开excel文件       打开工作表   跳过一行数据      读取并导入数据
+$fileObj->openFile($fileName)->openSheet()->setSkipRows(1)->importDataByCursor('insert_data',$pushHandle);
+
+//数据插入逻辑
+function insert_data($data)
+{
+    $db = DB::getInstance();
+    $sql = '';
+    foreach ($data as $v) {
+        $sql .= "(" . implode(",", $v) . "),";
+    }
+    $sql = trim($sql, ',');
+    $db->execute("insert into sheet2 (id,c1,c2,c3,c4) values " . $sql);
+}
+```

+ 9 - 9
examples/export_demo.php

@@ -65,14 +65,14 @@ $orderField =  [
 
 
 /**
 /**
  * 订单数据导出demo
  * 订单数据导出demo
- */
-$filePath = $fileObj
-    ->defaultFormat(['align'=>[Pxlswrite::FORMAT_ALIGN_CENTER,Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER]])//全局默认样式
-    ->field($orderField)//设置字段&表格头
-    ->setOrderData('generateOrderData',['order','time'],[],$pushHandle)
-    ->setColumn('A:B',50)
-    ->setColumn('F:F',50)
-    ->output();//输出excel文件到磁盘
+// */
+//$filePath = $fileObj
+//    ->defaultFormat(['align'=>[Pxlswrite::FORMAT_ALIGN_CENTER,Pxlswrite::FORMAT_ALIGN_VERTICAL_CENTER]])//全局默认样式
+//    ->field($orderField)//设置字段&表格头
+//    ->setOrderData('generateOrderData',['order','time'],[],$pushHandle)
+//    ->setColumn('A:B',50)
+//    ->setColumn('F:F',50)
+//    ->output();//输出excel文件到磁盘
 
 
 //单元格插入文本
 //单元格插入文本
 //for ($index = 0; $index < 10; $index++) {
 //for ($index = 0; $index < 10; $index++) {
@@ -128,7 +128,7 @@ function generateOrderData(){
         $range = mt_rand(0,5);
         $range = mt_rand(0,5);
         for($k = 0;$k<$range;$k++){
         for($k = 0;$k<$range;$k++){
             $orderId = date('YmdHis').rand(1000,9999);
             $orderId = date('YmdHis').rand(1000,9999);
-            $range2 = mt_rand(0,5);
+            $range2 = mt_rand(1,5);
             $item = [];
             $item = [];
             for($j = 0;$j<$range2;$j++){
             for($j = 0;$j<$range2;$j++){
                 $item[] = [
                 $item[] = [

+ 1 - 1
examples/import_demo.php

@@ -17,7 +17,7 @@ $fileName = end($fileInfo);
 //实例化WebSocketClient
 //实例化WebSocketClient
 $pushHandle = new WebSocketClient('ws://192.168.18.192:9502',$_GET['fd']);
 $pushHandle = new WebSocketClient('ws://192.168.18.192:9502',$_GET['fd']);
 //打开excel文件  setSkipRows(1)跳过一行数据
 //打开excel文件  setSkipRows(1)跳过一行数据
-$fileObj->openFile($fileName)->openSheet()->setSkipRows(1)->importData('insert_data',$pushHandle);
+$fileObj->openFile($fileName)->openSheet()->setSkipRows(1)->importDataByCursor('insert_data',$pushHandle);
 
 
 //数据插入逻辑
 //数据插入逻辑
 function insert_data($data)
 function insert_data($data)

+ 10 - 0
src/DataFormatException.php

@@ -0,0 +1,10 @@
+<?php
+
+
+namespace Pxlswrite;
+
+
+class DataFormatException extends \Exception
+{
+
+}

+ 10 - 0
src/PathException.php

@@ -0,0 +1,10 @@
+<?php
+
+
+namespace Pxlswrite;
+
+
+class PathException extends \Exception
+{
+
+}

+ 47 - 38
src/Pxlswrite.php

@@ -1,6 +1,6 @@
 <?php
 <?php
 /**
 /**
- * xlsxwriter简单封装
+ * xlswriter简单封装
  */
  */
 
 
 namespace Pxlswrite;
 namespace Pxlswrite;
@@ -97,7 +97,7 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 创建工作表
      * 创建工作表
-     * @param $_fileName
+     * @param string $_fileName
      * @param string $_tableName
      * @param string $_tableName
      * @return mixed
      * @return mixed
      */
      */
@@ -108,9 +108,13 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 设置字段
      * 设置字段
-     * @param $field
+     * @param array $field 字段定义数组 数据格式如下
+     * [
+     *  'name' => ['name' => '姓名','callback'=>'functionName'],
+     *  'age' => ['name' => '年龄'],
+     * ]
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function field($field)
     public function field($field)
     {
     {
@@ -125,15 +129,15 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 设置表格头
      * 设置表格头
-     * @param $header
-     * @param null $format_handle
+     * @param array $header
+     * @param resource|null $format_handle
      * @return mixed
      * @return mixed
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function header($header, $format_handle = NULL)
     public function header($header, $format_handle = NULL)
     {
     {
         if (count($header) !== count($header, 1)) {
         if (count($header) !== count($header, 1)) {
-            throw new \Exception('header数据格式错误,必须是一位数索引数组');
+            throw new DataFormatException('header数据格式错误,必须是一位数索引数组');
         }
         }
         $this->header = $header;
         $this->header = $header;
         if ($format_handle) {
         if ($format_handle) {
@@ -157,14 +161,14 @@ class Pxlswrite extends Excel
      * 通过生成器逐行向表格插入数据,
      * 通过生成器逐行向表格插入数据,
      * 设置过field才支持动态单元格合并,
      * 设置过field才支持动态单元格合并,
      * 可以根据指定的字段 通过值比较自动进行 行合并
      * 可以根据指定的字段 通过值比较自动进行 行合并
-     * @param $_generator 回调数据生成器方法 返回的数据格式是二维数组 如下字段名数量不限
+     * @param callable $_generator 回调数据生成器方法 返回的数据格式是二维数组 如下字段名数量不限
      * [['id'=>1,'name'=>'张三','age'=>'18']]
      * [['id'=>1,'name'=>'张三','age'=>'18']]
      * @param array $_mergeColumn 需要合并的字段
      * @param array $_mergeColumn 需要合并的字段
      * @param array $_mergeColumnStyle 合并单元格的样式
      * @param array $_mergeColumnStyle 合并单元格的样式
      * @param int $_index 单元格行偏移量 合并单元格的起始位置
      * @param int $_index 单元格行偏移量 合并单元格的起始位置
-     * @param WebSocketClient $_pushHandle
+     * @param WebSocketClient|null $_pushHandle
      * @return Pxlswrite
      * @return Pxlswrite
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function setDataByGenerator($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1)
     public function setDataByGenerator($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1)
     {
     {
@@ -230,7 +234,7 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 设置订单数据 根据数据可以合并指定的字段,需要遵循以下数据格式
      * 设置订单数据 根据数据可以合并指定的字段,需要遵循以下数据格式
-     * @param $_generator 数据生成器方法 返回数据格式如下,字段数量名称不限,只支持一个item二维数组
+     * @param callable $_generator 数据生成器方法 返回数据格式如下,字段数量名称不限,只支持一个item二维数组
      * [
      * [
      *    [
      *    [
      *        'order'=>'20200632555' ,
      *        'order'=>'20200632555' ,
@@ -252,7 +256,7 @@ class Pxlswrite extends Excel
      * @param WebSocketClient|null $_pushHandle WebSocketClient对象 用于推送进度
      * @param WebSocketClient|null $_pushHandle WebSocketClient对象 用于推送进度
      * @param int $_index 单元格行偏移量 合并单元格的起始位置
      * @param int $_index 单元格行偏移量 合并单元格的起始位置
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function setOrderData($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1)
     public function setOrderData($_generator, array $_mergeColumn = [], array $_mergeColumnStyle = [], WebSocketClient $_pushHandle = null, $_index = 1)
     {
     {
@@ -316,7 +320,7 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 字段过滤&格式化
      * 字段过滤&格式化
-     * @param $value 一维数组
+     * @param array $value 一维数组
      * @return array 处理之后的结果数组
      * @return array 处理之后的结果数组
      */
      */
     public function filter($value)
     public function filter($value)
@@ -333,12 +337,12 @@ class Pxlswrite extends Excel
     }
     }
 
 
     /**
     /**
-     * 导入数据
-     * @param $_func string 方法名 回调数据插入的方法
+     * 游标读取excel,分段插入数据库
+     * @param callable $_func 方法名 回调数据插入的方法
      * @param WebSocketClient|null $_pushHandle
      * @param WebSocketClient|null $_pushHandle
      * @param array $_dataType 可指定每个单元格数据类型进行读取
      * @param array $_dataType 可指定每个单元格数据类型进行读取
      */
      */
-    public function importData($_func, WebSocketClient $_pushHandle = null, array $_dataType = [])
+    public function importDataByCursor($_func, WebSocketClient $_pushHandle = null, array $_dataType = [])
     {
     {
         $count = 0;
         $count = 0;
         //游标读取excel数据 每一万条数据执行一次插入数据库 防止数据装载在内存过大
         //游标读取excel数据 每一万条数据执行一次插入数据库 防止数据装载在内存过大
@@ -378,15 +382,15 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 文件下载
      * 文件下载
-     * @param $_filePath 文件绝对路径
+     * @param string $_filePath 文件绝对路径
      * @param bool $_isDelete 下载后是否删除原文件
      * @param bool $_isDelete 下载后是否删除原文件
-     * @throws \Exception
+     * @throws PathException
      */
      */
     public function download($_filePath, $_isDelete = true)
     public function download($_filePath, $_isDelete = true)
     {
     {
 //        setcookie("loadingFlag",1);
 //        setcookie("loadingFlag",1);
         if (dirname($_filePath) != $this->m_config['path']) {
         if (dirname($_filePath) != $this->m_config['path']) {
-            throw new \Exception('未知文件路径');
+            throw new PathException('未知文件路径:'.dirname($_filePath));
         }
         }
         header("Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
         header("Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
         header('Content-Disposition: attachment;filename="' . end(explode('/', $_filePath)) . '"');
         header('Content-Disposition: attachment;filename="' . end(explode('/', $_filePath)) . '"');
@@ -409,7 +413,7 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 打开文件
      * 打开文件
-     * @param $zs_file_name 文件名称
+     * @param string $zs_file_name 文件名称
      * @return mixed
      * @return mixed
      */
      */
     public function openFile($zs_file_name)
     public function openFile($zs_file_name)
@@ -419,7 +423,7 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 读取表格
      * 读取表格
-     * @param $_fileName
+     * @param string $_fileName
      * @return mixed
      * @return mixed
      */
      */
     public function import($_fileName)
     public function import($_fileName)
@@ -431,6 +435,11 @@ class Pxlswrite extends Excel
         return $data;
         return $data;
     }
     }
 
 
+    /**
+     * 写日志
+     * @param string $_message
+     * @param array $_arr
+     */
     public function writeLog($_message, array $_arr)
     public function writeLog($_message, array $_arr)
     {
     {
         $dir = rtrim($this->m_config['path'], '/') . '/log/';
         $dir = rtrim($this->m_config['path'], '/') . '/log/';
@@ -445,7 +454,7 @@ class Pxlswrite extends Excel
      * 格式化样式
      * 格式化样式
      * @param array $_style 样式列表数组
      * @param array $_style 样式列表数组
      * @return Format resource
      * @return Format resource
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function styleFormat($_style)
     public function styleFormat($_style)
     {
     {
@@ -455,7 +464,7 @@ class Pxlswrite extends Excel
             switch ($key) {
             switch ($key) {
                 case 'align':
                 case 'align':
                     if (!is_array($value) || count($value) != 2) {
                     if (!is_array($value) || count($value) != 2) {
-                        throw new \Exception('align 数据格式错误');
+                        throw new DataFormatException('align 数据格式错误');
                     }
                     }
                     $format->align($value[0], $value[1]);
                     $format->align($value[0], $value[1]);
                     break;
                     break;
@@ -474,11 +483,11 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 行单元格样式
      * 行单元格样式
-     * @param $range string 单元格范围
-     * @param $height double 单元格高度
-     * @param null $formatHandler resource|array 单元格样式
+     * @param string $range  单元格范围
+     * @param double $height 单元格高度
+     * @param resource|array $formatHandler  单元格样式
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function setRow($range, $height, $formatHandler = null)
     public function setRow($range, $height, $formatHandler = null)
     {
     {
@@ -491,11 +500,11 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 列单元格样式
      * 列单元格样式
-     * @param $range string 单元格范围
-     * @param $width double 单元格宽度
-     * @param null $formatHandler resource|array 单元格样式
+     * @param string $range 单元格范围
+     * @param double $width 单元格宽度
+     * @param resource|array $formatHandler 单元格样式
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function setColumn($range, $width, $formatHandler = null)
     public function setColumn($range, $width, $formatHandler = null)
     {
     {
@@ -508,11 +517,11 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 合并单元格
      * 合并单元格
-     * @param $scope  string 单元格范围
-     * @param $data   string data
-     * @param null $formatHandler resource|array style
+     * @param string $scope   单元格范围
+     * @param string $data    data
+     * @param resource|array $formatHandler  合并单元格的样式
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function mergeCells($scope, $data, $formatHandler = null)
     public function mergeCells($scope, $data, $formatHandler = null)
     {
     {
@@ -525,9 +534,9 @@ class Pxlswrite extends Excel
 
 
     /**
     /**
      * 全局默认样式
      * 全局默认样式
-     * @param $formatHandler resource|array style
+     * @param resource|array $formatHandler style
      * @return $this
      * @return $this
-     * @throws \Exception
+     * @throws DataFormatException
      */
      */
     public function defaultFormat($formatHandler)
     public function defaultFormat($formatHandler)
     {
     {