java 動(dòng)態(tài)生成bean的案例
最近做一個(gè)需求,需求中的bean只用于生成一次json使用,所以想通過配置來動(dòng)態(tài)的生成,查了一下,java還真有這個(gè)實(shí)現(xiàn)。
java動(dòng)態(tài)的生成javabean,只能生成屬性和對(duì)應(yīng)的set/get方法,不能生成其他的方法。
import org.assertj.core.internal.cglib.beans.BeanGenerator;
import org.assertj.core.internal.cglib.beans.BeanMap;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
/**
* Created by wangpengzhi1 on 2018/1/2.
*/
public class BeanCeater {
public static void main(String[] args) throws ClassNotFoundException {
System.out.println("Generate JavaBean");
Map properties = new HashMap();
properties.put("id", Class.forName("java.lang.Integer"));
properties.put("name", Class.forName("java.lang.String"));
properties.put("address", Class.forName("java.lang.String"));
Object stu = generateObject(properties);
System.out.println("Set values");
setValue(stu, "id", 123);
setValue(stu, "name", "454");
setValue(stu, "address", "789");
System.out.println("Get values");
System.out.println(">> " + getValue(stu, "id"));
System.out.println(">> " + getValue(stu, "name"));
System.out.println(">> " + getValue(stu, "address"));
System.out.println("Show all methods");
Method[] methods = stu.getClass().getDeclaredMethods();
for(Method method : methods) {
System.out.println(">> " + method.getName());
}
System.out.println("Show all properties");
Field[] fields = stu.getClass().getDeclaredFields();
for(Field field : fields) {
System.out.println(">> " + field.getName());
}
}
private static Object generateObject(Map properties) {
BeanGenerator generator = new BeanGenerator();
Set keySet = properties.keySet();
for(Iterator i = keySet.iterator(); i.hasNext();) {
String key = (String)i.next();
generator.addProperty(key, (Class)properties.get(key));
}
return generator.create();
}
private static Object getValue(Object obj, String property) {
BeanMap beanMap = BeanMap.create(obj);
return beanMap.get(property);
}
private static void setValue(Object obj, String property, Object value) {
BeanMap beanMap = BeanMap.create(obj);
beanMap.put(property, value);
}
}
代碼不難懂,有需要的自己復(fù)制。
補(bǔ)充:spring 工具類 ReflectionUtils 獲取bean所有字段
以前遇到要獲取當(dāng)前類以及所有父類的的field的時(shí)候,都是遞歸一直往上找,一直到Object ,個(gè)人覺得這種方法是不是太low了,有沒有更好的辦法?或者jdk其實(shí)是有這種方法的,只是我不知道,今天看了下spring中的實(shí)現(xiàn),也是一樣的,真沒有更好的辦法?
public static void doWithFields(Class<?> clazz, ReflectionUtils.FieldCallback fc, @Nullable ReflectionUtils.FieldFilter ff) {
Class targetClass = clazz;
do {
Field[] fields = getDeclaredFields(targetClass);
Field[] var5 = fields;
int var6 = fields.length;
for(int var7 = 0; var7 < var6; ++var7) {
Field field = var5[var7];
if (ff == null || ff.matches(field)) {
try {
fc.doWith(field);
} catch (IllegalAccessException var10) {
throw new IllegalStateException("Not allowed to access field '" + field.getName() + "': " + var10);
}
}
}
targetClass = targetClass.getSuperclass();
} while(targetClass != null && targetClass != Object.class);
}
private static Field[] getDeclaredFields(Class<?> clazz) {
Assert.notNull(clazz, "Class must not be null");
Field[] result = (Field[])declaredFieldsCache.get(clazz);
if (result == null) {
try {
result = clazz.getDeclaredFields();
declaredFieldsCache.put(clazz, result.length == 0 ? NO_FIELDS : result);
} catch (Throwable var3) {
throw new IllegalStateException("Failed to introspect Class [" + clazz.getName() + "] from ClassLoader [" + clazz.getClassLoader() + "]", var3);
}
}
return result;
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
- 如何動(dòng)態(tài)修改JavaBean中注解的參數(shù)值
- Java cglib為實(shí)體類(javabean)動(dòng)態(tài)添加屬性方式
- java通過cglib動(dòng)態(tài)生成實(shí)體bean的操作
- Java 如何從spring容器中獲取注入的bean對(duì)象
- JavaBean實(shí)體類處理外鍵過程解析
- 通過實(shí)例解析POJO和JavaBean的區(qū)別
- java Beanutils.copyProperties( )用法詳解
- AndroidStudio插件GsonFormat之Json快速轉(zhuǎn)換JavaBean教程
- JavaBean valication驗(yàn)證實(shí)現(xiàn)方法示例
- java中PO、VO、BO、POJO、DAO、DTO、TO、QO、Bean、conn的理解
- Java 確保某個(gè)Bean類被最后執(zhí)行的幾種實(shí)現(xiàn)方式
相關(guān)文章
Java Socket編程筆記_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
Socket對(duì)于我們來說就非常實(shí)用了。下面是本次學(xué)習(xí)的筆記。主要分異常類型、交互原理、Socket、ServerSocket、多線程這幾個(gè)方面闡述2017-05-05
java中List數(shù)組用逗號(hào)分隔開轉(zhuǎn)成字符串2種方法
在我們?nèi)粘i_發(fā)中,在前后端交互的時(shí)候會(huì)遇到多個(gè)id或其他字段存放到一個(gè)字段中,這時(shí)我們會(huì)遇到一個(gè)List(集合)---->String(單個(gè)字段),這篇文章主要給大家介紹了關(guān)于java中List數(shù)組用逗號(hào)分隔開轉(zhuǎn)成字符串的2種方法,需要的朋友可以參考下2023-10-10
spring注解之@Valid和@Validated的區(qū)分總結(jié)
@Validated和@Valid在基本驗(yàn)證功能上沒有太多區(qū)別,但在分組、注解地方、嵌套驗(yàn)證等功能上有所不同,下面這篇文章主要給大家介紹了關(guān)于spring注解之@Valid和@Validated區(qū)分的相關(guān)資料,需要的朋友可以參考下2022-03-03
Java中的CountDownLatch簡(jiǎn)單理解
這篇文章主要介紹了Java中的CountDownLatch簡(jiǎn)單理解,CountDownLatch是一個(gè)同步工具類,用來攜調(diào)多個(gè)線程之間的同步,它是是使用一個(gè)計(jì)數(shù)器進(jìn)行實(shí)現(xiàn)的,計(jì)數(shù)器初始值為線程數(shù)量,需要的朋友可以參考下2024-01-01
Java設(shè)計(jì)模式之備忘錄模式_動(dòng)力節(jié)點(diǎn)Java學(xué)院
我們?cè)诰幊痰臅r(shí)候,經(jīng)常需要保存對(duì)象的中間狀態(tài),當(dāng)需要的時(shí)候,可以恢復(fù)到這個(gè)狀態(tài)。接下來通過本文給大家分享java設(shè)計(jì)模式之備忘錄模式,感興趣的的朋友一起看看吧2017-08-08
Spring boot項(xiàng)目部署到云服務(wù)器小白教程詳解
這篇文章主要介紹了Spring boot項(xiàng)目部署到云服務(wù)器小白教程詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-04-04

