EasyExcel相关

打印 上一主题 下一主题

主题 778|帖子 778|积分 2334

1. easyexcel–100M

EasyExcel是一个基于Java的使用简单、节省内存的读写Excel的开源项目。在尽大概节省内存的环境下支持读写百M的Excel。
节省内存的原因:在剖析Excel时没有将文件数据一次性全部加载到内存中,而是从磁盘上一行行读取数据,逐个剖析
EasyExcel接纳一行一行的剖析模式,并将一行的剖析结果以观察者的模式通知处理惩罚
   通过Java代码完成对Excel的读写操作。所谓的读写理解为上传和下载
  github地点:https://github.com/alibaba/easyexcel
  官方文档:https://www.yuque.com/easyexcel/doc
  2. easyexcel写操作

   所谓的写操作,就是把Java中的类对象写入到excel表格中
  

实现步调

  • 引入依靠
  1. <dependency>
  2.             <groupId>repMaven.com.alibaba</groupId>
  3.             <artifactId>easyexcel</artifactId>
  4.             <version>3.0.5</version>
  5.         </dependency>
复制代码

  • 封装相应的对象,创建与表格对应的实体类
    1. @Data
    2. @AllArgsConstructor
    3. @NoArgsConstructor
    4. public class ExcelDemo {
    5.     @ExcelProperty(value = "姓名")//标记excel的表头内容
    6.     public String name;
    7.     @ExcelProperty(value = "年龄")
    8.     public Integer age;
    9.     @ExcelProperty(value = "性别")
    10.     public String sex;
    11.     @ExcelIgnore//写入excel表格时忽略该属性
    12.     public String address;
    13. }
    复制代码

    • @ExcelProperty:标志excel的表头内容
    • @ExcelIgnore:在写入excel表格时忽略该属性
       
  • 通过easyexcel完成写入操作
    1. public class WriteExcel {
    2.     public static void main(String[] args) {
    3.         //fileName:表示excel写入的路径以及名称
    4.         String fileName="D:\\idea\\easyexcel_demo1\\writer.xlsx";
    5.         //这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板,然后文件流会自动关闭
    6.         //如果这里使用03,则 传入 excelType 参数即可
    7.         List<ExcelDemo> list=new ArrayList<>();
    8.         list.add(new ExcelDemo("章三",18,"男","郑州"));
    9.         list.add(new ExcelDemo("李斯",19,"男","郑州"));
    10.         list.add(new ExcelDemo("汪芜",20,"女","郑州"));
    11.         list.add(new ExcelDemo("码字",25,"男","郑州"));
    12.         EasyExcel.write(fileName,ExcelDemo.class).sheet("第一次完成写操作").doWrite(list);
    13.     }
    14. }
    复制代码

    • 预备文件路径
    • 写出文件
       
3. easyexcel写操作——web模式【导出】

导出:必要将数据库内里的文件以附件的形式下载到本地电脑,必要参数为response对象,返回值类型为void


  • 实例
  1. @Controller
  2. public class ExcelController {
  3.     @GetMapping("download")
  4.     public void download(HttpServletResponse response) throws IOException {
  5.         // 这里注意 有同学反应使用swagger 会导致各种问题,请直接用浏览器或者用postman
  6.         response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
  7.         response.setCharacterEncoding("utf-8");
  8.         // 这里URLEncoder.encode可以防止中文乱码 当然和easyexcel没有关系
  9.         String fileName = URLEncoder.encode("测试", "UTF-8").replaceAll("\\+", "%20");
  10.         response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName + ".xlsx");
  11.         List<ExcelDemo> list=new ArrayList<>();
  12.         list.add(new ExcelDemo("章三",18,"男","郑州"));
  13.         list.add(new ExcelDemo("李斯",19,"男","郑州"));
  14.         list.add(new ExcelDemo("汪芜",20,"女","郑州"));
  15.         list.add(new ExcelDemo("码字",25,"男","郑州"));
  16.         EasyExcel.write(response.getOutputStream(), ExcelDemo.class).sheet("模板").doWrite(list);
  17.     }
  18. }
复制代码
也可以在controllre层仅进行调用操作,将逻辑交由service去做,即:


  • controller
  1.     @ApiOperation("导出")
  2.     @GetMapping("/download")
  3.     public void exportData(HttpServletResponse response){
  4.         dictService.exportData(response);
  5.     }
复制代码


  • service
  1. @Override
  2.     public void exportData(HttpServletResponse response) {
  3.         try {
  4.             //设置相关参数
  5.             response.setContentType("application/vnd.ms-excel");
  6.             response.setCharacterEncoding("utf-8");
  7.             // 这里URLEncoder.encode可以防止中文乱码 当然和easyexcel没有关系
  8.             String fileName = URLEncoder.encode("数据字典", "UTF-8");
  9.             response.setHeader("Content-disposition", "attachment;filename="+ fileName + ".xlsx");
  10.             //获取文件
  11.             List<Dict> list = this.baseMapper.selectList(null);
  12.             //转换文件
  13.             ArrayList<DictEeVo> dictEeVos = new ArrayList<>();
  14.             for (Dict dict : list) {
  15.                 DictEeVo dictEeVo = new DictEeVo();
  16.                 //转换
  17.                 BeanUtils.copyProperties(dict, dictEeVo);
  18.                 //添加
  19.                 dictEeVos.add(dictEeVo);
  20.             }
  21.             //写出
  22.               EasyExcel.write(response.getOutputStream(), DictEeVo.class).sheet("数据字典").doWrite(dictEeVos);
  23.           } catch (Exception e) {
  24.               e.printStackTrace();
  25.         }
  26.     }
复制代码
测试
测试仅需答应项目,然后在前端页面写入现实的url地点即可,也可以添加按钮的点击事件进行测试
4. easyexcel读操作



  • 流程



  • 实现步调

  • 引入fastjson依靠
    1. <dependency>
    2.             <groupId>com.alibaba</groupId>
    3.             <artifactId>fastjson</artifactId>
    4.             <version>1.2.83</version>
    5.         </dependency>
    复制代码
          com.alibaba.fastjson.JSON是阿里巴巴开源的一个高性能数据交换格式库,它是基于Java语言的JSON剖析和生成工具。Fastjson旨在提供一种快速、便捷的方式来序列化和反序列化Java对象到JSON字符串,以及将JSON字符串转换回Java对象。它支持大数据量的处理惩罚,而且具有优秀的性能,通常比尺度的Java自带的java.util.JSONObject和org.json.JSONObject更快。
        使用Fastjson的好处包括:

    • 高效:通过字节码操作,使得JSON剖析和生成的速度非常快。
    • 易用:API设计简洁明了,易于理解和上手。
    • 功能丰富:支持复杂数据结构的处理惩罚,如日期、数组、聚集等。
        在Java项目中,你可以通过Maven或Gradle添加fastjson依靠来使用它,然后通过JSONObject、JSONArray或其他类来进行JSON相关的操作。
       
  • 创建监听器
    1. package com.zmq.excel;
    2. import com.alibaba.excel.context.AnalysisContext;
    3. import com.alibaba.excel.read.listener.ReadListener;
    4. import com.alibaba.excel.util.ListUtils;
    5. import com.alibaba.fastjson.JSON;
    6. import lombok.extern.slf4j.Slf4j;
    7. import java.util.List;
    8. // 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
    9. @Slf4j
    10. public class DemoDataListener implements ReadListener<ExcelDemo> {
    11.     /**
    12.      * 每隔5条存储数据库,实际使用中可以100条,然后清理list ,方便内存回收
    13.      */
    14.     private static final int BATCH_COUNT = 100;
    15.     /**
    16.      * 缓存的数据
    17.      */
    18.     private List<ExcelDemo> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
    19.     /**
    20.      * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。
    21.      */
    22.     private DemoDAO demoDAO;
    23.     public DemoDataListener() {
    24.         // 这里是demo,所以随便new一个。实际使用如果到了spring,请使用下面的有参构造函数
    25.         demoDAO = new DemoDAO();
    26.     }
    27.     /**
    28.      * 如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来
    29.      *
    30.      * @param demoDAO
    31.      */
    32.     public DemoDataListener(DemoDAO demoDAO) {
    33.         this.demoDAO = demoDAO;
    34.     }
    35.     /**
    36.      * 这个每一条数据解析都会来调用
    37.      *
    38.      * @param data    one row value. Is is same as {@link AnalysisContext#readRowHolder()}
    39.      * @param context
    40.      */
    41.     //一行一行的去读取里面的数据
    42.     @Override
    43.     public void invoke(ExcelDemo data, AnalysisContext context) {
    44.         log.info("解析到一条数据:{}", JSON.toJSONString(data));
    45.         cachedDataList.add(data);
    46.         // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM
    47.         if (cachedDataList.size() >= BATCH_COUNT) {
    48.             saveData();
    49.             // 存储完成清理 list
    50.             cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
    51.         }
    52.     }
    53.     /**
    54.      * 所有数据解析完成了 都会来调用
    55.      *
    56.      * @param context
    57.      */
    58.     @Override
    59.     public void doAfterAllAnalysed(AnalysisContext context) {
    60.         // 这里也要保存数据,确保最后遗留的数据也存储到数据库
    61.         saveData();
    62.         log.info("所有数据解析完成!");
    63.     }
    64.     /**
    65.      * 加上存储数据库
    66.      */
    67.     private void saveData() {
    68.         log.info("{}条数据,开始存储数据库!", cachedDataList.size());
    69.         demoDAO.save(cachedDataList);
    70.         log.info("存储数据库成功!");
    71.     }
    72. }
    复制代码
          进行读操作时就会触发监听器,基本不必要改动
  • dao
    1. package com.zmq.excel;
    2. import java.util.List;
    3. /**
    4. * 假设这个是你的DAO存储。当然还要这个类让spring管理,当然你不用需要存储,也不需要这个类。
    5. **/
    6. public class DemoDAO {
    7.     public void save(List<ExcelDemo> list) {
    8.         // 如果是mybatis,尽量别直接调用多次insert,自己写一个mapper里面新增一个方法batchInsert,所有数据一次性插入
    9.     }
    10. }
    复制代码
          dao中填写必要的操作
  • 读取
    1. public class ReadExcel {
    2.     public static void main(String[] args) {
    3.         /**
    4.          * 指定列的下标或者列名
    5.          *
    6.          * <p>1. 创建excel对应的实体对象,并使用{@link ExcelProperty}注解. 参照{@link IndexOrNameData}
    7.          * <p>2. 由于默认一行行的读取excel,所以需要创建excel一行一行的回调监听器,参照{@link IndexOrNameDataListener}
    8.          * <p>3. 直接读即可
    9.          */
    10.         //fileName:表示excel读取的路径以及名称
    11.         String fileName="D:\\idea\\easyexcel_demo1\\writer.xlsx";
    12.         // 这里默认读取第一个sheet
    13.         //new DemoDataListener():监听器
    14.         EasyExcel.read(fileName, ExcelDemo.class, new DemoDataListener()).sheet().doRead();
    15.     }
    16. }
    复制代码
5. easyexcel文件上传-web【导入】

导入:必要将本地文件插入到数据库,参数:multiparefile,返回值:“成功或失败”
   使用excel进行导入必要剖析器的共同,使用监听器对读取的文件进行操作
  剖析器:用来读取文件,并将数据插入到数据库
  

  • 参加文件上传的依靠
    1. <dependency>
    2.             <groupId>repMaven.commons-fileupload</groupId>
    3.             <artifactId>commons-fileupload</artifactId>
    4.             <version>1.4</version>
    5.         </dependency>
    复制代码
          commons-fileupload是一个开源库,专为Java应用程序设计,用于处理惩罚文件上传功能。它简化了处理惩罚HTTP哀求中的multipart/form-data格式,这种格式通常用于用户通过Web表单上传文件到服务器。该库提供了一组工具类和处理惩罚器,使得开辟者能够剖析上传的文件流,并管理文件名、大小限制等复杂操作。它的API易于使用,支持断点续传以及错误处理惩罚,广泛应用于web应用开辟中的文件上传组件。
  • 文件上传剖析器——spring设置文件中
    若使用springboot框架,仅引入上述依靠即可
    1. <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
    2.         <property name="maxUploadSize" value="1000000000"/>
    3.     </bean>
    复制代码
  • controller
    1. @Autowired
    2.     private DemoDAO demoDAO;
    3.     @PostMapping("/upload")
    4.     @ResponseBody //将返回值转化为json格式
    5.     public String upload(MultipartFile file) throws IOException{
    6.         EasyExcel.read(file.getInputStream(), ExcelDemo.class, new DemoDataListener(demoDAO)).sheet().doRead();
    7.         return "success";
    8.     }
    复制代码
          DemoDao层参加注入spring容器的注解:@Component
  • 测试——使用postman完成测试


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

何小豆儿在此

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表