java注解之運(yùn)行時(shí)修改字段的注解值操作
今天遇到需求:導(dǎo)入Excel時(shí)候列頭會(huì)發(fā)生變化,客戶是大爺要求你改代碼,
導(dǎo)入Excel是用easypoi做的,識(shí)別表頭是用注解@Excel(name = "xxx")通過這個(gè)name來匹配
那你表頭要?jiǎng)?,我這個(gè)注解是硬編碼
所以就有動(dòng)態(tài)設(shè)置這個(gè)表頭
public class JavaVo{
@Excel(name = "xxx")
private String userName;
//省略getset方法
}
ExcelImportUtil.importExcel(file.getInputStream(), configClass(JavaVo.class), params);
代碼如下
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");
// 因?yàn)檫@個(gè)字段事 private final 修飾,所以要打開權(quán)限
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;
}
補(bǔ)充知識(shí):java動(dòng)態(tài)修改 注解的值,控制對象轉(zhuǎn)化為json字符串的字段是否序列化
定義一個(gè)對象使用@JSONField控制該對象屬性是否需要序列化
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;
}
編寫工具類
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;
/**
* 動(dòng)態(tài)操作注解屬性
* @since 2020年8月13日20:49:26
*/
public class AnnotationUtils<T> {
/**
* 查看注解屬性
* @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;
}
}
測試
import com.alibaba.fastjson.JSON;
public class TT {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
AnnotationUtils<A> 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里面的注解看看效果,這個(gè)好像是發(fā)生了jvm優(yōu)化導(dǎo)致的問題。。。
注釋第一個(gè)print 打印結(jié)果如下:
false
true
{"extendParams":"exex","sad":"sadsad"}
不注釋第一個(gè)print 打印結(jié)果如下:
false
{"sad":"sadsad"}
true
{"sad":"sadsad"}
接下來我們在做一個(gè)測試
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
List<A> aList = new ArrayList<>();
for(int i=0; i<10; i++){
AnnotationUtils<A> 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));
}
打印結(jié)果
[{"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"}]
我本想用修改注解的方式來修改某個(gè)字段的序列化與不序列化,但是我發(fā)現(xiàn)注解是在class層面的并不是在對象層面。所以我的設(shè)想失敗了。。
以上這篇java注解之運(yùn)行時(shí)修改字段的注解值操作就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- java注解結(jié)合aspectj AOP進(jìn)行日志打印的操作
- 如何動(dòng)態(tài)修改JavaBean中注解的參數(shù)值
- java使用@Scheduled注解執(zhí)行定時(shí)任務(wù)
- Java中l(wèi)ombok的@Builder注解的解析與簡單使用詳解
- java 注解默認(rèn)值操作
- 解決IDEA導(dǎo)入javaWeb項(xiàng)目注解爆紅的問題
- javax.validation自定義日期范圍校驗(yàn)注解操作
- 談?wù)凧ava中自定義注解及使用場景
- Java如何使用JSR303校驗(yàn)數(shù)據(jù)與自定義校驗(yàn)注解
- java SpringBoot自定義注解,及自定義解析器實(shí)現(xiàn)對象自動(dòng)注入操作
- Java 注解學(xué)習(xí)筆記
相關(guān)文章
Spring Security 構(gòu)建rest服務(wù)實(shí)現(xiàn)rememberme 記住我功能
這篇文章主要介紹了Spring Security 構(gòu)建rest服務(wù)實(shí)現(xiàn)rememberme 記住我功能,需要的朋友可以參考下2018-03-03
關(guān)于使用ContextClassLoader遇到的問題
這篇文章主要介紹了關(guān)于使用ContextClassLoader遇到的問題,ContextClassLoader是通過Thread.currentThread().getContextClassLoader()返回該線程上下文的ClassLoader,需要的朋友可以參考下2023-10-10
Springboot啟動(dòng)后執(zhí)行方法小結(jié)
本文主要介紹了Springboot啟動(dòng)后執(zhí)行方法小結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04
關(guān)于Gateway網(wǎng)關(guān)中配置跨域的三種方案
文章總結(jié):介紹了三種處理跨域請求的方法:在Controller類上添加注解、通過配置類實(shí)現(xiàn)重寫WebMvcConfigurer接口和在配置文件中統(tǒng)一設(shè)置,希望這些方法能幫助讀者解決跨域問題2024-11-11

