• 使用Reflect封装Excel导出工具类


    本文使用的emoji: 💧 💦 ☔

    最近遇到一个需求,需要对页面的列表数据做导出操作,考虑了很多实现方案之后,最终选择了hutool的ExcelWriter + Spring自带的Reflet包来实现这个功能,采用这种方式的主要原因是对现有代码改动较少,并且可以无缝切入系统,实现各个模块的导出操作,设计思路如下:

    如上图所示,首先我们定义一个接口,此接口完成Excel导出功能,在ExcelExportService 中实现功能。

    话不多说,上代码:

    💧定义ExcelExportEnum

    在导出Excel之前,我们需要先定义ExcelExportEnum,此枚举存储了我们后端需要导出的模块和模板名称

    1. @Getter
    2. @RequiredArgsConstructor
    3. public enum ExcelExportEnum {
    4. /**
    5. * 模块
    6. */
    7. MODULE_A("moduleAService", "模块A");
    8. private final String key;
    9. private final String name;
    10. }
    11. 复制代码

    💧定义导出方法

    💦校验入参

    首先,我们需要对入参进行校验,查看传入的 MODULE_NAME 是否和后端Enum的一致,并且我们需要对参数中的自定义方法名进行提取,我们通过ExcelExportEnum来获取导出的module的Service名称

    1. if (!map.containsKey(MODULE_NAME)) {
    2. throw new BaseException("缺少参数:moduleName");
    3. }
    4. String moduleName = map.get(MODULE_NAME).toString();
    5. if (!EnumUtil.contains(ExcelExportEnum.class, moduleName)) {
    6. throw new BaseException("模块查找失败");
    7. }
    8. String functionName = DEFAULT_FUNCTION_NAME;
    9. //如果传了自定义方法名
    10. if (map.containsKey(CUSTOM_FUNCTION_NAME)) {
    11. functionName = map.get(CUSTOM_FUNCTION_NAME).toString();
    12. }
    13. String serviceName = ExcelExportEnum.valueOf(moduleName).getKey();
    14. String tableName = ExcelExportEnum.valueOf(moduleName).getName();
    15. 复制代码

    💦利用java反射获取Service中的所有method

    1. Map<String, Method> methodMap = Arrays.stream(SpringContextHolder.getBean(serviceName).getClass().getMethods()).collect(Collectors.toMap(Method::getName, Function.identity(), (key1, key2) -> key2));
    2. 复制代码

    在上述代码中SpringContextHolder.getBean(serviceName).getClass().getMethods()方法,首先通过getBean来获取 第一步中得到的ServiceName的Java bean。然后通过Class.getMethods()方法获取此service中的所有方法

    最后,使用Arrays.stream方法,将module对应的Service中的所有method获取到,并放入methodMap中,供后续代码使用

    💦提取method中的入参对象,获取其Class

    1. Object o;
    2. try {
    3. o = methodMap.get(functionName).getParameterTypes()[0].getDeclaredConstructor().newInstance();
    4. } catch (Exception e) {
    5. throw new BaseException(e);
    6. }
    7. 复制代码

    通过Method.getParameterTypes()方法,将第二步中获取的method的入参的Class取到,并赋值给对象Object o,供后续调用分页列表接口传递入参匹配Class使用

    💦传入参数,调用导出方法

    1. Map<String, Object> param = (Map<String, Object>) map.computeIfAbsent(PARAM, k -> new HashMap<String, Object>(4));
    2. param.put("limit", -1);
    3. CopyOptions copyOptions = new CopyOptions();
    4. copyOptions.setIgnoreError(true);
    5. PageUtils page = (PageUtils) methodMap.get(functionName).invoke(SpringContextHolder.getBean(serviceName), BeanUtil.mapToBean(param, o.getClass(), true, copyOptions));
    6. if (CollUtil.isEmpty(page.getList())) {
    7. throw new BaseException("数据获取失败");
    8. }
    9. Object resultObject = page.getList().get(0);
    10. ExcelUtils utils = new ExcelUtils(resultObject.getClass());
    11. utils.exportExcel(page.getList(), response, tableName, moduleName);
    12. 复制代码

    可以看到,在上述导出方法中,我们使用了Method.invoke方法,并且设置转义忽略

    💧注意事项

    使用hutool工具包输出到流

    1. // 通过工具类创建writer,默认创建xls格式
    2. ExcelWriter writer = ExcelUtil.getWriter();
    3. //创建xlsx格式的
    4. //ExcelWriter writer = ExcelUtil.getWriter(true);
    5. // 一次性写出内容,使用默认样式,强制输出标题
    6. writer.write(rows, true);
    7. //out为OutputStream,需要写出到的目标流
    8. writer.flush(out);
    9. // 关闭writer,释放内存
    10. writer.close();
    11. 复制代码

    注意 ExcelUtil.getWriter()默认创建xls格式的Excel,因此写出到客户端也需要自定义文件名为XXX.xls,否则会出现文件损坏的提示。 若想生成xlsx格式,请使用ExcelUtil.getWriter(true)创建。

     

  • 相关阅读:
    JS-语法-变量(声明、命名规范、一次性声明多个变量、使用)
    Matlab:设置命令历史记录预设项
    安防监控/视频存储/视频汇聚平台EasyCVR如何接入智能分析网关V4?
    【Web前端开发 规范 参考手册】
    【PDN仿真笔记7-使用Sigrity PowerSI进行PDN仿真的方法2】
    Json“牵手”易贝商品详情数据方法,易贝商品详情API接口,易贝API申请指南
    FMT固定翼无人机目标搜寻与投掷
    PyG (PyTorch Geometric) 异质图神经网络HGNN
    手把手怎么把照片修复高清,p图小白也能轻松上手
    双系统安装教程(win10 & ubuntu20.04)
  • 原文地址:https://blog.csdn.net/BASK2311/article/details/127819334