基于 EasyExcel实现用户自定义导出excel顺序

基于 EasyExcel实现用户自定义导出excel顺序

pom 文件

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>easyexcel</artifactId>
    <version>3.3.2</version>
</dependency>

实体类

@Data
public class Student {
        private static final long serialVersionUID = 1L;
        @ExcelProperty(value = "名字")
     	private String name;
        @ExcelProperty(value = "年龄")
        private String age;
    
 }

控制层

@PostMapping("/export")
public void export(HttpServletResponse response, @RequestBody Student stu) {
    //用户选择要导出的循序,这里模拟前端传递
    List<String> includeColumnFiledNames = new ArrayList<>();
    includeColumnFiledNames.add("name");
	includeColumnFiledNames.add("age");
    String fileName = "学生一览"
    //查询数据库
    List<Student> data = studentSereice.getstuList(stu);
   	//导出
    ExcelUtils.write(response, fileName + ".xlsx", fileName, Student.class, data, includeColumnFiledNames);

}

工具类

public class ExcelUtils {
	private ExcelUtils() {

	}
    	/**
	 * 将列表以 Excel 响应给前端
	 * @param response  响应
	 * @param fileName  文件名
	 * @param sheetName Excel sheet 名
	 * @param head      Excel head 头
	 * @param data      数据列表哦
	 * @param <T>       泛型,保证 head 和 data 类型的一致性
	 * @throws IOException 写入失败的情况
	 */
	public static <T> void write(HttpServletResponse response, String fileName, String sheetName, Class<T> head, List<T> data,
								 List<String> includeColumnFiledNames) throws Exception {
//设置响应头,指定为Excel文件下载
response.addHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));
		response.setContentType("application/vnd.ms-excel;charset=UTF-8");

EasyExcelFactory.write(response.getOutputStream(), head).
    			// 只要导出对象中的字段的数据
				includeColumnFieldNames(includeColumnFiledNames)
    			//会根据传入集合的顺序排序
				.orderByIncludeColumn(true)
				.registerWriteHandler(new LongestMatchColumnWidthStyleStrategy())
				.sheet(sheetName)
				.doWrite(data);
	}
}

3.3版本以下的坑

​ orderByIncludeColumn 方法是3.3新增方法,includeColumnFieldNames方法只是筛选并不负责排序,所以正常思路是根据前端传递的includeColumnFiledNames 参数动态设置导出实体类的属性ExcelProperty注解的index值.如下:

private static <T> void handelHead(Class<T> head, List<String> includeColumnFiledNames) throws NoSuchFieldException, IllegalAccessException {
		for (String filedName  :includeColumnFiledNames){
			Field field = head.getDeclaredField(filedName);
			ExcelProperty excelPropertyAnnotation = field.getAnnotation(ExcelProperty.class);
			InvocationHandler h = Proxy.getInvocationHandler(excelPropertyAnnotation);
			Field indexField = h.getClass().getDeclaredField("memberValues");
			// 因为这个字段事 private final 修饰,所以要打开权限
			indexField.setAccessible(true);
			// 获取 memberValues
			Map<String,Object> memberValues =  (Map<String, Object>) indexField.get(h);

			// 修改 value 属性值
			memberValues.put("index", includeColumnFiledNames.indexOf(filedName));
		}
}

以上代码第一次运行是没问题的,但用户如果重新选择新的顺序就会出第一次运行的顺序.看了官方文档才知道解析class的field会有缓存,以前全局放到Map里面,3.3.0 以后默认放到ThreadLocal里了.所以建议使用3.3版本,或不使用实体类导出,使用List<Map<String,Object>>

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
EasyExcel是一个基于Java的简单易用的Excel文件读写工具,它可以很方便地实现Excel文件导入和导出。下面介绍一下如何使用EasyExcel实现Excel文件导出。 1. 引入EasyExcel依赖 将以下依赖添加到项目的pom.xml文件中: ```xml <dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>2.1.7</version> </dependency> ``` 2. 创建Excel实体类 首先,需要创建一个Java类来表示Excel文件的数据结构。例如,如果要导出一个包含学生信息的Excel文件,可以创建一个名为Student的类,并在其中定义需要导出的属性: ```java public class Student { private String name; private Integer age; private String gender; private String address; // 省略getter和setter方法 } ``` 3. 创建Excel文件 在进行Excel文件导出之前,需要先创建一个Excel文件。EasyExcel提供了一个名为ExcelWriter的类,可以用来创建Excel文件。以下是一个创建Excel文件的示例代码: ```java // 创建ExcelWriter对象 ExcelWriter excelWriter = new ExcelWriter(new FileOutputStream("students.xlsx"), ExcelTypeEnum.XLSX); // 创建Sheet对象 Sheet sheet = new Sheet(1, 0, Student.class); // 设置Sheet名称 sheet.setSheetName("学生信息"); // 写入数据 List<Student> data = new ArrayList<>(); // 添加数据到List中 // ... excelWriter.write(data, sheet); // 关闭ExcelWriter对象 excelWriter.finish(); ``` 以上代码中,创建了一个名为“学生信息”的Sheet对象,并设置其数据类型为Student类。然后,将需要导出的数据写入到Excel文件中。 4. 导出Excel文件 创建Excel文件后,就可以通过EasyExcel将其导出了。以下是一个导出Excel文件的示例代码: ```java // 设置响应头信息 response.setContentType("application/vnd.ms-excel"); response.setCharacterEncoding("utf-8"); response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("students.xlsx", "UTF-8")); // 创建ExcelWriter对象 ExcelWriter excelWriter = new ExcelWriter(response.getOutputStream(), ExcelTypeEnum.XLSX, true); // 创建Sheet对象 Sheet sheet = new Sheet(1, 0, Student.class); // 设置Sheet名称 sheet.setSheetName("学生信息"); // 写入数据 List<Student> data = new ArrayList<>(); // 添加数据到List中 // ... excelWriter.write(data, sheet); // 关闭ExcelWriter对象 excelWriter.finish(); ``` 以上代码中,设置了响应头信息,告诉浏览器需要下载一个Excel文件。然后,创建了一个ExcelWriter对象,并将其输出流设置为响应流,这样Excel文件就可以直接输出到浏览器中。最后,将需要导出的数据写入到Excel文件中,并关闭ExcelWriter对象。 通过以上步骤,就可以使用EasyExcel实现Excel文件导出了。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值