如何设计一个通用的 Excel 导入导出功能?

失败的面 老炮说Java 2024-04-07 14:00 山西

老炮说Java

十年老炮程序员带你玩技术

公众号

以JSON配置的方式去实现通用性和动态调整,当然,这个通用仍然存在一定的局限性,每个项目的代码风格都不同,想要写出一个适合所有项目的通用性模块并不容易,这里的通用局限于其所在项目,所以该功能代码如果不适用于自己的项目,希望可以以此为参考,稍作修改。

那么现在来分析一下,我们会需要哪些JSON配置项。

导出

基础配置项

先从最简单的导出开始,被导出数据应该支持通过业务层查出,如:Service.search(param),这是大前提,然后为了支持显示导出进度,业务层还需要提供数量查询方法,如:Service.count(param),否则无法实现导出进度。

最后导出文件名也可以定制,如:filename

由上可以得出配置项:

  • serviceClazz: 业务类路径,如:com.cc.service.UserService,必填

  • methodName: 查询方法名,如:listByCondition,必填

  • countMethodName: 数量查询方法名,可填,用于支持导出进度

  • filename: 导出文件名

  • searchParams: 查询参数,数组类型,字典元素。用数组是为了支持查询方法需要传多参数的情况

至于查询方法的参数类,不需要填,因为我们可以通过反射去获取到该方法所需要传入的参数类型(注意,以下贴出的是关键代码,仅作参考理解):

Class<?> serviceClass = Class.forName(param.getServiceClazz());

// param为请求参数类
Method searchMethod = ReflectUtil.findMethodByName(serviceClass, param.getMethodName());

// 方法所需要传入的参数列表
Class<?>[] parameterTypes = searchMethod.getParameterTypes();
/**
 * 通过反射从指定类中获取方法对象
 */
public static Method findMethodByName(Class<?> clazz, String name) {
    Method[] methods = clazz.getMethods();
    if (StringUtils.isEmpty(name)) {
        return null;
    }
    for (Method method : methods) {
        if (method.getName().equals(name)) {
            return method;
        }
    }
    return null;
}

现在我们来想想,导出都会有哪些场景:

  • 列表页的分页查询,可能是当前页数据导出,也可能是所有数据导出,这涉及到分页查询

  • 数据总览页的查询,通常是开发者自定义的复杂连表查询,不需要分页

那么本文针对以上两种情况来实现第一版的通用导出功能。

列表页的分页查询

列表页的数据导出分当前页导出和所有数据导出,假设查询流程是这样的:

  • 接口层接收参数:Controller.search(Param param)

  • 业务层调用查询方法:Service.search(param)

  • 持久层访问数据库:Mapper.search(param)

这种情况很简单,但如果流程是这样的:

  • 接口层接收参数:Controller.search(Param param)

  • 业务层调用查询方法:Service.search(new Condition(param))

  • 持久层访问数据库:Mapper.search(condition)

上面代码中,接口请求参数和持久层参数不一致,在业务层经过了包装,那么这种情况也要兼容处理。

但是如果请求参数在业务层经过了包中包中包,那么就算了。

接着是分页参数,我们用pageNum和pageSize来表示页码和数量字段,类似于:

{
    "pageNum": 1,
    "pageSize": 10,
    "name": "老刘" // 此为查询字段,如查询名字为老刘的数据
}

关于当前页导出和所有数据导出,可以用一个bool来表示:onlyCurrentPage,默认false,即导出时会自动分页查询数据,直到所有数据查询完毕,导出所有数据时分页查询很有必要,能提高性能,避免内存溢出,当onlyCurrentPage为true时,则只导出当前页面数据。

得出需要的配置项为:

  • searchParam: 接口分页请求参数,JSON类型,必填

  • conditionClazz: 条件查询类,也可以认为是包装类,如:com.cc.codition.UserCondition,可填

  • onlyCurrentPage: 仅当前页导出,默认false,可填

数据总览页的查询

数据总览数据没有数量查询方法,即Service.count(xxx),也没有分页查询参数,类似于当前页导出,在也只考虑一层包装类的情况下,没有额外的配置项,上面的已经足够了,要注意的就是代码里面得把分页参数剔除掉。

表头配置

一级表头

模拟一些数据来加深理解,现有一个接口是查询系统用户列表,如:/user/search,返回结果是这样的:

{
    "code": 0,
    "msg": "请求成功",
    "data": [
        {
            "id": 1,
            "username": "admin",
            "nickname": "超管",
            "phone": "18818881888",
            "createTime": "2023-06-23 17:16:00"
        },
        {
            "id": 2,
            "username": "cc",
            "nickname": "管理员",
            "phone": "18818881888",
            "createTime": "2023-06-23 17:16:00"
        },
        ...
    ]
}

现在贴出EasyExcel的代码:

// 创建excel文件
try (ExcelWriter excelWriter = EasyExcel.write(path).build()) {
    WriteSheet writeSheet = EasyExcel.writerSheet("sheet索引", "sheet名称").head(getHeader()).build();
    excelWriter.write(getDataList(), writeSheet);
}
// 模拟表头
private static List<List<String>> getHeader() {
    List<List<String>> list = new ArrayList<>();
    list.add(createHead("账号"));
    list.add(createHead("昵称"));
    list.add(createHead("联系方式"));
    list.add(createHead("注册时间"));

    return list;
}

public static List<String> createHead(String... head) {
    return new ArrayList<>(Arrays.asList(head));
}

// 模拟数据
public static List<List<Object>> getDataList() {
    List<List<Object>> list = new ArrayList<>();
    list.add(createData("admin", "超管", "18818881888", "2023-06-23 17:16:00"));
    list.add(createData("cc", "管理员", "18818881888", "2023-06-23 17:16:00"));
    return list;
}

public static List<Object> createData(String... data) {
    return new ArrayList<>(Arrays.asList(data));
}

然后导出效果是这样的:

图片

现在先别在乎效果图的excel样式,我们后面都会进行动态配置,比如列宽、表头背景色、字体居中等。

上面我们虽然是写死了代码,但聪明的开发者一定懂得将数据库查询来的数据转换成对应的格式,所以这段就跳过了。

现在我们就可以得出基础的表头配置:

"customHeads": [
    {
        "fieldName": "username", 
        "fieldNameZh": "账号"
    },
    {
        "fieldName": "nickname",
        "fieldNameZh": "昵称"
    },
    {
        "fieldName": "phone",
        "fieldNameZh": "联系方式"
    },
    {
        "fieldName": "createTime",
        "fieldNameZh": "注册时间"
    }
]

也就是:

  • fieldName: 属性名,这样可以从返回结果的数据对象里面通过反射找到该属性以及值

  • fieldNameZh: 属性名肯定不适合作为表头名,增加一个中文说明来代替属性名作为表头

有了上面的基础,我们就可以增加更多的项来实现功能的丰富性,比如

{
    "fieldName": "username", 
    "fieldNameZh": "账号",
    "width": 20, // 列宽
    "backgroundColor": 1, // 表头背景色
    "fontSize": 20, // 字体大小
    "type": "date(yyyy-MM-dd)" // 字段类型
    ...
}

注:字段类型可以用作数据格式化,比如该属性是一个status状态,1表示正常,2表示异常,那么导出这个1或2是没有意义的,所以通过字段类型识别出这个状态值对应的中文描述,这样的导出才正常。

一级表头已经可以满足我们许多场景了,但是这并不足够,我的经验中,经常需要用到两行表头甚至是复杂表头,好在EasyExcel是支持多级表头的。

多级表头

先贴出EasyExcel生成二级表头的示例代码:

// 模拟表头
private static List<List<String>> getHeader() {
    List<List<String>> list = new ArrayList<>();
    list.add(createHead("用户信息", "账号"));
    list.add(createHead("用户信息", "昵称"));
    list.add(createHead("用户信息", "联系方式"));
    list.add(createHead("用户信息", "注册时间"));

    list.add(createHead("角色信息", "超管"));
    list.add(createHead("角色信息", "管理员"));

    return list;
}

public static List<String> createHead(String... head) {
    return new ArrayList<>(Arrays.asList(head));
}

// 模拟数据
public static List<List<Object>> getDataList() {
    List<List<Object>> list = new ArrayList<>();
    list.add(createData("admin", "超管", "18818881888", "2023-06-23 17:16:00", "是", "是"));
    list.add(createData("cc", "管理员", "18818881888", "2023-06-23 17:16:00", "否", "是"));
    return list;
}

public static List<Object> createData(String... data) {
    return new ArrayList<>(Arrays.asList(data));
}

效果是这样的:

图片

可以看到,前面4列有一个共同表头【用户信息】,后面两列有一个共同表头【角色信息】,从上面的示例代码我们知道,要使表头合并,数据列表得按顺序和相同表头名,这样会被EasyExcel识别到然后才有合并效果,这点需要注意。

同理,当我们需要生成复杂表头的时候,可以这样:

// 模拟表头
private static List<List<String>> getHeader() {
    List<List<String>> list = new ArrayList<>();
    list.add(createHead("导出用户数据", "用户信息", "账号"));
    list.add(createHead("导出用户数据", "用户信息", "昵称"));
    list.add(createHead("导出用户数据", "用户信息", "联系方式"));
    list.add(createHead("导出用户数据", "用户信息", "注册时间"));

    list.add(createHead("导出用户数据", "角色信息", "超管"));
    list.add(createHead("导出用户数据", "角色信息", "管理员"));

    return list;
}

效果图:

图片

结论

以上是我对导出功能的思考和实现思路,因为篇幅的关系,我没有贴出完整的代码,但是相信以上内容已经足够大家作为参考,缺少的内容,比如列宽、颜色字体等设置,请查阅EasyExcel官方文档来实现,主要方式就是根据前端传过来的JSON配置信息,来动态配置EasyExcel的导出文件。

导入

导入分两个步骤:

  • 用户下载导入模板

  • 用户填内容进导入模板,然后上传模板文件到系统,实现数据导入操作

下载导入模板

导入模板只需要上面的customHeads参数即可:

"customHeads": [
        {
            "fieldName": "username", 
            "fieldNameZh": "账号"
        },
        {
            "fieldName": "nickname",
            "fieldNameZh": "昵称"
        },
        {
            "fieldName": "phone",
            "fieldNameZh": "联系方式"
        },
        {
            "fieldName": "createTime",
            "fieldNameZh": "注册时间"
        }
    ]

甚至fieldName都可以不要,生成一个只有表头的excel文件。

导入数据

导入数据有两种场景:

  • 单表数据导入,该场景很简单

  • 复杂数据导入,涉及多表,这种情况就稍微复杂点

单表数据导入

单表只需要考虑对应实体类的属性即可,我们可以通过反射来获取实体类的属性,所以需要的配置项是:

  • modelClazz: 实体类路径,如:com.cc.entity.User

配置示例:

{
    "modelClazz": "com.cc.entity.User",
    "customHeads": [
        {
            "fieldName": "username", 
            "fieldNameZh": "账号"
        },
        {
            "fieldName": "nickname",
            "fieldNameZh": "昵称"
        },
        {
            "fieldName": "phone",
            "fieldNameZh": "联系方式"
        },
        {
            "fieldName": "createTime",
            "fieldNameZh": "注册时间"
        }
    ]
}

这样在导入数据,被EasyExcel读取每一行数据的时候,可以识别到如:username项对应com.cc.entity.User类的username属性那么就能做到类似这样的事情:

User user = new User();
user.setUsername(fieldName列的值)

由此可以得到一个List<User> userList数组,再通过系统的UserService或UserMapper保存到数据库,即可实现数据导入操作。

复杂数据导入

复杂数据比如这种场景:excel文件中每行的数据是这样的:

图片

其中是否超管和是否管理员涉及关联表:

  • 用户表:tb_user

  • 角色表:tb_role

  • 用户角色关联表:tb_user_role_relation

为了支持这种复杂数据导入,系统内需要提供对应的保存方法:

1.新建DTO类:

第一种:

public class UserDto {
    private String username;
    private String nickname;
    private String phone;
    private Date createTime;
    private Boolean superAdminFlag;
    private Boolean adminFlag;
}

第二种:

public class UserDto {
    private User user;
    private Role role;
}

这两种DTO的情况我们都应该考虑,第一种不用多说,上面的配置就可以应对,主要看第二种,第二种方式要考虑“路径”这个问题,所以customHeads的写法就要有所改变:

{
    "modelClazz": "com.cc.model.UserDto",
    "customHeads": [
        {
            "fieldName": "user.username", 
            "fieldNameZh": "账号"
        },
        ...
    ]
}

这样配置账号路径为:user.username,属性的反射查询就要有递归概念,先去查找UserDto类的user属性,得到该属性的类,再去获取其内的username属性,赋值方式就变成了:

UserDto dto = new UserDto();
User user = new User();
user.setUsername(fieldName列的值);
dto.setUser(user);

这样得到一个List<UserDto> dtoList数组。

2.既然有复杂数据导入的业务,那么在Service业务层中,也应该编写复杂数据的保存函数:

public interface UserService {
    // 单条插入
 void saveUserDto(UserDto dto);
    
    // 批量插入
    void saveUserDtoBatch(List<UserDto> dtoList);
}
@Service
public class UserServiceImpl implements UserService {
    @Autowired
    private UserMapper userMapper;
    
    @Autowired
    private RoleService roleService;
    
    @Autowired
    private UserRoleRelationService relationService;
    
    // 事务
    @Transactional(rollbackFor = Exception.class)
    @Override
    public void saveUserDto(UserDto dto) {
        // 保存用户
        User user = userMapper.save(dto.getUser());
        // 保存角色
        Role role = roleService.save(dto.getRole);
        // 保存关联
        UserRoleRelation relation = new UserRoleRelation();
        relation.setUserId(user.getId());
        relation.setRoleId(role.getId());
        relationService.save(relation);
    }
    
    // 批量插入代码省略,原理同上
    void saveUserDtoBatch(List<UserDto> dtoList);
}

3.通过EasyExcel读取到的每一行数据都能转成UserDto对象,再通过单条或批量来保存数据,这期间有许多可以优化考虑的点,比如:

  • 批量比单条保存效率高、性能好,但是批量不容易识别出部分失败的行

  • 批量保存的数量不能太多,要考虑系统和数据库的性能,比如每次读取500行就执行一次保存

  • 保存的进度显示,先获取excel总行数,再根据当前读取行数来计算进度,并返回给前端

  • 导入时间过长,可以做成后台任务进行,至于前端提醒可以是轮询也可以是WebSocket

所以需要指定查询方法,这配置项上面已经给出来了。

配置项总结

最后给出一个总配置项出来参考:

导出数据配置

{
    "filename": "用户数据导出",
    "serviceClazz": "com.cc.service.UserService",
    "methodName": "listByCondition",
    "countMethodName": "countByCondition",
    "searchParams": [
        {
            "nickname": "cc" // 搜索昵称为cc的用户
        }
    ],
    "customHeads": [
        {
          "fieldName": "username",
            "fieldNameZh": "账号",
            "width": 20, // 列宽
            "fontSize": 20 // 字体大小
        },
        {
            "fieldName": "createTime",
            "fieldNameZh": "注册时间",
            "type": "date(yyyy-MM-dd)" // 属性类型声明为date,并且转换成指定格式导出
        }
    ]
}

导入模板配置

{
    "filename": "用户数据导入",
    "modelClazz": "com.cc.entity.User",
    "customHeads": [
        {
          "fieldName": "username",
            "fieldNameZh": "账号",
            "width": 20, // 列宽
            "fontSize": 20 // 字体大小
        },
        {
            "fieldName": "createTime",
            "fieldNameZh": "注册时间",
            "type": "date(yyyy-MM-dd)" // 属性类型声明为date,并且转换成指定格式导出
        }
    ]
}

导入数据配置

{
    "modelClazz": "com.cc.entity.User",
    "serviceClazz": "com.cc.service.UserService",
    "methodName": "save",
    "customHeads": [
        {
          "fieldName": "username",
            "fieldNameZh": "账号",
        },
        {
            "fieldName": "createTime",
            "fieldNameZh": "注册时间",
            "type": "date(yyyy-MM-dd)" // 属性类型声明为date,并且转换成指定格式导出
        }
    ]
}

来源:juejin.cn/post/7248281795937697852

 

资料免费共享群

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java是一种通用的高级编程语言,通过Java编写程序可以实现各种功能,包括Excel导入导出。在Java中,可以使用Apache POI来操作Excel文件,Apache POI是一个流行的用于读写Microsoft Office文件的Java库,它包括HSSF(用于读写Excel 97-2003),XSSF(用于读写Excel 2007+)和SS(实验性的通用Spiral包)。 在Java中使用Apache POI进行Excel导入时,首先需要创建一个Workbook对象,该对象可以是XSSFWorkbook(用于.xlsx文件格式),也可以是HSSFWorkbook(用于.xls文件格式)。一旦获得了Workbook对象,就可以从工作表中获取数据。对于每个工作表,都可以使用Sheet对象访问单元格中的值,并使用Row对象访问单元格中的列数据。 在Excel导出方面,在Java中使用Apache POI可以创建Excel文档并向其中添加数据。要创建一个Excel文档,首先需要创建一个Workbook对象,然后添加一个或多个工作表。对于每个工作表,可以使用Sheet对象访问单元格,并使用Row对象访问列数据。将数据添加到单元格后,可以使用FileOutputStream将文档写入指定的文件。 总的来说,使用Java进行Excel导入导出是一种非常灵活和高效的方法,可以通过学习和使用Apache POI等工具来实现。 ### 回答2: Java一个跨平台的编程语言,它可以很好地进行Excel文件的导入导出Java提供了一些库,比如Apache POI,用于操作Excel文件。通过这些库,我们可以在Java中进行Excel文件的读取和写入。 Excel导入导出可以用来处理各种数据,这对于企业来说非常有用。因为数据是企业的核心资产之一,所以正确导入导出数据是非常重要的。Java对于处理Excel数据非常方便,我们可以很容易地将数据转换成Java对象,然后以任何我们需要的方式进行处理。 在使用Java进行Excel导入导出时,我们需要先导入Apache POI库。通过这个库,我们可以调用其提供的方法来操作Excel文件。例如,我们可以通过创建Workbook对象来读取Excel文件,并通过Sheet对象访问工作表中的单元格。 对于Excel导出,我们首先需要定义Excel模板,然后使用Java代码填充模板。我们可以通过创建Workbook对象来创建新的工作簿。然后,我们可以创建一个Sheet对象,然后使用Row对象和Cell对象来创建表格。最后,我们将表格数据写入Excel文件中。 总而言之,Java提供了快速,可靠的Excel导入导出解决方案。这对于企业来说非常重要,因为数据是企业运营的核心资产。利用JavaExcel导入导出功能,企业可以更好地处理数据并提高效率。 ### 回答3: Java导入导出Excel文档是非常常见的操作,可以根据具体需求使用Java自带的API或者开源框架完成。本文将重点介绍Apache POI和EasyExcel两个开源框架的使用。 一、Apache POI Apache POI是Apache软件基金会的一个开源项目,提供基于Java的API用于处理Microsoft Office格式文件。其中包括Word、Excel和PowerPoint等文档的读写操作,对于Excel的读写操作使用的是XSSF和HSSF两个模块,分别支持.xlsx和.xls格式的文件。 1.导入Excel 首先需要导入POI的jar包,然后使用以下代码读取Excel中的数据: ``` File file = new File("test.xlsx"); InputStream inputStream = new FileInputStream(file); Workbook workbook = new XSSFWorkbook(inputStream);//读取xlsx格式 //Workbook workbook = new HSSFWorkbook(inputStream);//读取xls格式 Sheet sheet = workbook.getSheetAt(0);//获取第一个sheet Iterator<Row> iterator = sheet.iterator(); while(iterator.hasNext()) { Row row = iterator.next(); Iterator<Cell> cellIterator = row.cellIterator(); while(cellIterator .hasNext()) { Cell cell = cellIterator.next(); switch (cell.getCellType()) { case STRING: System.out.print(cell.getStringCellValue() + " "); break; case NUMERIC: System.out.print(cell.getNumericCellValue() + " "); break; case BOOLEAN: System.out.print(cell.getBooleanCellValue() + " "); break; case FORMULA: System.out.print(cell.getCellFormula() + " "); break; default: System.out.print(""); } } System.out.println();//换行 } inputStream.close(); ``` 2.导出Excel 基于POI的Excel导出主要是通过创建Workbook对象、Sheet对象和Cell对象,然后将数据填充到Cell对象中,最后输出到文件中。以下是导出Excel的示例代码: ``` Workbook workbook = new XSSFWorkbook();//创建工作簿 Sheet sheet = workbook.createSheet("sheet0");//创建sheet Row row = sheet.createRow(0);//创建行 Cell cell = row.createCell(0);//创建单元格 cell.setCellValue("姓名"); cell = row.createCell(1); cell.setCellValue("年龄"); cell = row.createCell(2); cell.setCellValue("性别"); row = sheet.createRow(1);//创建行 cell = row.createCell(0); cell.setCellValue("张三"); cell = row.createCell(1); cell.setCellValue(20); cell = row.createCell(2); cell.setCellValue("男"); FileOutputStream outputStream = new FileOutputStream("test.xlsx");//输出到文件 workbook.write(outputStream); outputStream.close(); ``` 二、EasyExcel EasyExcel一个基于POI封装的Java开源框架,提供简单易用的方式完成Excel导入导出操作。相对于POI,EasyExcel的优点在于它的API更加简单明了,使用起来更加方便快捷。 1.导入Excel EasyExcel导入Excel的方式主要是通过读取监听器实现,可以实现不同sheet的读取、数据转换和对数据的处理等功能。以下是导入Excel的示例代码: ``` //创建读取监听器 public class ExcelListener extends AnalysisEventListener<User> { private List<User> userList = new ArrayList<>(); @Override public void invoke(User user, AnalysisContext analysisContext) { userList.add(user);//每读取一条数据就添加到List中 } @Override public void doAfterAllAnalysed(AnalysisContext analysisContext) { //读取完成后的操作 } public List<User> getUserList() { return userList; } public void setUserList(List<User> userList) { this.userList = userList; } } //使用监听器读取Excel数据 File file = new File("test.xlsx"); EasyExcel.read(file, User.class, new ExcelListener()).sheet().doRead(); ``` 2.导出Excel EasyExcel导出Excel的方式主要是通过生成器模式创建写入对象、构造sheet页和单元格,最后输出到文件或者流中。以下是导出Excel的示例代码: ``` //创建写入对象 ExcelWriter writer = EasyExcel.write("test.xlsx", User.class).build(); //构造sheet页和单元格 WriteSheet sheet = EasyExcel.writerSheet("sheet0").build(); //填充数据到单元格 List<User> userList = new ArrayList<>(); userList.add(new User("张三", 20, "男")); userList.add(new User("李四", 22, "女")); writer.write(userList, sheet); //输出到文件或者流中 writer.finish(); ``` 总结 以上是Apache POI和EasyExcel两个框架的Excel导入导出操作,两者都十分流行和实用,可以根据项目需求进行选择。在实际开发中,需要注意的是Excel的版本问题、数据类型的转换、异常处理等,并加以优化和完善,从而提高代码的可维护性和效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值