反射获取对象成员的字段值,getFields()和getDeclaredFields()用法区别

24 篇文章 1 订阅
4 篇文章 0 订阅
  用反射获取内部类的属性其实很简单。。我弄了半天才弄好,由于很弱智的原因啊,写此博文已吸取教训。
  虽简单,但是太坑爹了,getFields()只能获取public的字段,包括父类的。
  而getDeclaredFields()只能获取自己声明的各种字段,包括public,protected,private。
  而我写的Characters类中的属性是在继承父类的,父类中是protected的,所以获取不到,这个弄了我半天!最后只要把父类的protected属性全改成public的就ok了啊。。

  还有getFields()和 getDeclaredFields(),返回的都是Field对象,获取名称直接field.getName(), 但是属性值则是field.get(Object),这个object是该field所属的!!!

  太坑爹了,绕了好多弯啊!!!不过这样却对反射获得字段有了深刻的理解。
  具体代码如下:
/**
	 * 生成java实体类的核心方法
	 * @param clazz
	 * @param targetDir
	 * @throws Exception 
	 */
	private void genFileHandler(Class<?> clazz, String targetDir) throws Exception {
		/* 1,获取模板 */
		Template temp = configuration.getTemplate("testEntity.ftl", pageEncoding);
		temp.setEncoding(pageEncoding);
		
		/* 2,设置模板的数据内容Map */
		// 获取相关名称
		String entityName = getEntityName(clazz);
		String enumName = clazz.getSimpleName();
		String tableName = getTableName(clazz);
		
		// 填充Map
		Map<String, Object> targetClazz = new HashMap<String, Object>();
		Map<String, Map<String, String>> fields = new HashMap<String, Map<String, String>>();
		targetClazz.put("entityName", entityName);
		targetClazz.put("enumName", enumName);
		targetClazz.put("tableName", tableName);
		targetClazz.put("fields", fields);
		
		// 获得所有枚举字段成员(id, account, name, profession...),
		// 遍历每个枚举成员,获取属性,放入Map中
		Object[] enums = clazz.getEnumConstants();
		for (Object e : enums) {
			Map<String, String> field = new HashMap<String, String>();
			String name = e.toString();
			Class<?> type = (Class<?>) getFieldValue(clazz, e, "type");
			
			field.put("name", name);
			field.put("type", type.getSimpleName());
			fields.put(e.toString() + "field", field);
		}
		
		// 判断目标文件夹不存在,则新建文件夹
		File dir = new File(targetDir);
		if(!dir.exists())	dir.mkdirs();
		
		/* 3,将模板和数据合并,并生成文件 */
		String fileName = targetDir + entityName + ".java";
		System.out.println("-------开始生成" + fileName + "文件......------");
		
		File target = new File(fileName);
		Writer out = new OutputStreamWriter(new FileOutputStream(target), pageEncoding);
		temp.process(targetClazz, out);
		out.flush();
		out.close();

		System.out.println("-------" + fileName + "文件生成完毕!-------");
	}




还有其父类部分代码改过之后是:
public abstract class EntityConfCommon {
	public Class<?> type;
	public int length;
	public String index;
	public String defaults;
	public boolean isNull;

main方法是:
	public static void main(String[] args) throws SecurityException, IllegalArgumentException, NoSuchFieldException, IllegalAccessException, InstantiationException {
		Class<Character> c = Character.class;

		FreemarkerGenEntity generator = new FreemarkerGenEntity();
		generator.init();
		generator.genFile(c, "d:\\test\\");
		
	}

这样就没问题了!
Java 反射可以通过以下方式获取父类的字段: ```java Class<?> superClass = targetClass.getSuperclass(); // 获取目标类的父类 Field[] fields = superClass.getDeclaredFields(); // 获取父类的所有字段 ``` 其中,`targetClass` 是目标类的 `Class` 对象,`superClass` 是目标类的父类的 `Class` 对象,`fields` 是父类的所有字段数组。使用 `getDeclaredFields()` 可以获取所有的字段,包括私有字段。如果只需要获取公有字段,则可以使用 `getFields()` 方法。如果需要获取指定名称的字段,则可以使用 `getDeclaredField(String name)` 方法或 `getField(String name)` 方法。 需要注意的是,如果父类的字段是私有的,则需要通过 `setAccessible(true)` 方法来设置其可见性,否则在获取时会抛出 `IllegalAccessException` 异常。完整代码如下: ```java public static void main(String[] args) { ChildClass child = new ChildClass(); Class<?> targetClass = child.getClass(); Class<?> superClass = targetClass.getSuperclass(); // 获取目标类的父类 Field[] fields = superClass.getDeclaredFields(); // 获取父类的所有字段 for (Field field : fields) { field.setAccessible(true); // 设置私有字段可见 try { System.out.println(field.getName() + " = " + field.get(child)); // 获取并输出字段 } catch (IllegalAccessException e) { e.printStackTrace(); } } } public static class ParentClass { private String parentField = "parentValue"; } public static class ChildClass extends ParentClass { private String childField = "childValue"; } ``` 输出结果为: ``` parentField = parentValue ```
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值