最近做一个需求,需求中的bean只用于生成一次json使用,所以想通过配置来动态的生成,查了一下,java还真有这个实现。
java动态的生成javabean,只能生成属性和对应的set/get方法,不能生成其他的方法。
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
47
48
49
50
51
52
53
54
55
56
|
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);
}
}
|
代码不难懂,有需要的自己复制。
补充:spring 工具类 ReflectionUtils 获取bean所有字段
以前遇到要获取当前类以及所有父类的的field的时候,都是递归一直往上找,一直到Object ,个人觉得这种方法是不是太low了,有没有更好的办法?或者jdk其实是有这种方法的,只是我不知道,今天看了下spring中的实现,也是一样的,真没有更好的办法?
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
|
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;
}
|
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。如有错误或未考虑完全的地方,望不吝赐教。
原文链接:https://blog.csdn.net/wangpengzhi19891223/article/details/78949572