今天遇到需求:导入Excel时候列头会发生变化,客户是大爷要求你改代码,
导入Excel是用easypoi做的,识别表头是用注解@Excel(name = "xxx")通过这个name来匹配
那你表头要动,我这个注解是硬编码
所以就有动态设置这个表头
1
2
3
4
5
6
7
public class JavaVo{
@Excel(name = "xxx")
private String userName;
//省略getset方法
}
ExcelImportUtil.importExcel(file.getInputStream(), configClass(JavaVo.class), params);
代码如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
private Class configClass(Class c , String val) {
Field[] fields = c.getDeclaredFields();
try {
for(int i = 0;i < fields.length;i++){
Field f = fields[i];
Excel excelAn = f.getAnnotation(Excel.class);//Excel是注解类型
if(excelAn == null){
continue;
}
InvocationHandler h = Proxy.getInvocationHandler(excelAn);
Field hField = h.getClass().getDeclaredField("memberValues");
// 因为这个字段事 private final 修饰,所以要打开权限
hField.setAccessible(true);
// 获取 memberValues
Map memberValues = (Map) hField.get(h);
// 修改 value 属性值 这里修改的是@Excel(name = "姓名")
//name是key
memberValues.put("name", val);
}
} catch (Exception e) {
e.printStackTrace();
}
return c;
}
补充知识:java动态修改 注解的值,控制对象转化为json字符串的字段是否序列化
定义一个对象使用@JSONField控制该对象属性是否需要序列化
1
2
3
4
5
6
7
8
9
10
11
import com.alibaba.fastjson.annotation.JSONField;
import lombok.Data;
@Data
public class A {
@JSONField(serialize = false)
private String extendParams;
@JSONField(serialize = true)
private String sad;
}
编写工具类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
import com.alibaba.fastjson.annotation.JSONField;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.util.Map;
import lombok.val;
/**
* 动态操作注解属性
* @since 2020年8月13日20:49:26
*/
public class AnnotationUtils
/**
* 查看注解属性
* @param t
* @param name
* @return
* @throws NoSuchFieldException
*/
public Object getJSONFieldProp(T t, String name) throws NoSuchFieldException {
Field field = t.getClass().getDeclaredField(name);
JSONField annotation = field.getAnnotation(JSONField.class);
val serialize = annotation.serialize();
return serialize;
}
/**
* 修改注解属性
* @param t
* @param value
* @return
* @throws NoSuchFieldException
* @throws IllegalAccessException
*/
public Object setJSONFieldProp(T t,String name, Object value) throws NoSuchFieldException, IllegalAccessException {
Field field = t.getClass().getDeclaredField(name);
JSONField annotation = field.getAnnotation(JSONField.class);
InvocationHandler invocationHandler = Proxy.getInvocationHandler(annotation);
Field memberValues = invocationHandler.getClass().getDeclaredField("memberValues");
memberValues.setAccessible(true);
Map map = (Map) memberValues.get(invocationHandler);
map.put("serialize",value);
val serialize = annotation.serialize();
return serialize;
}
}
测试
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import com.alibaba.fastjson.JSON;
public class TT {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
AnnotationUtils aAnnotationUtils = new AnnotationUtils<>();
A a = new A();
a.setExtendParams("exex");
a.setSad("sadsad");
Object extendParams = aAnnotationUtils.getJSONFieldProp(a, "extendParams");//查询注解的值
System.out.println(extendParams.toString());
// System.out.println(JSON.toJSONString(a));
Object extendParams1 = aAnnotationUtils.setJSONFieldProp(a, "extendParams", true);//修改注解的值
System.out.println(extendParams1.toString());
System.out.println(JSON.toJSONString(a));
}
}
去掉main里面的注解看看效果,这个好像是发生了jvm优化导致的问题。。。
注释第一个print 打印结果如下:
1
2
3
false
true
{"extendParams":"exex","sad":"sadsad"}
不注释第一个print 打印结果如下:
1
2
3
4
false
{"sad":"sadsad"}
true
{"sad":"sadsad"}
接下来我们在做一个测试
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
List aList = new ArrayList<>();
for(int i=0; i<10; i++){
AnnotationUtils aAnnotationUtils = new AnnotationUtils<>();
A a = new A();
a.setExtendParams("exex");
a.setSad("sadsad");
if(i%2 == 0) {
aAnnotationUtils.setJSONFieldProp(a, "extendParams", true);//修改注解的值
}
aList.add(a);
}
System.out.println(JSON.toJSONString(aList));
}
打印结果
1
[{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"},{"extendParams":"exex","sad":"sadsad"}]
我本想用修改注解的方式来修改某个字段的序列化与不序列化,但是我发现注解是在class层面的并不是在对象层面。所以我的设想失败了。。