反射机制:getDeclaredField和getField的区别

在做后台开发时实体的固定字段一般会抽取为一个父类,其他类继承该父类,例如主键字段,会放到一个父类中(IdEntity),其他类继承该类,但是我们在操作时操作的是子类,在通过子类获取父类属性是getDeclaredField和getField是不一样的,简单说

  1. getDeclaredFiled 仅能获取类本身的属性成员(包括私有、共有、保护) 
  2. getField 仅能获取类(及其父类可以自己测试) public属性成员

因此在获取父类的私有属性时,要通过getSuperclass的之后再通过getDeclaredFiled获取。

父类,出去id作为父类字段

import javax.persistence.Column;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.MappedSuperclass;

import org.hibernate.annotations.GenericGenerator;
@MappedSuperclass
public abstract class IdEntity {
	private String id;

	@Id
	@GeneratedValue(generator = "paymentableGenerator")
	@GenericGenerator(name = "paymentableGenerator", strategy = "uuid")
	@Column(name ="ID",nullable=false,length=32)
	public String getId() {
		return id;
	}

	public void setId(String id) {
		this.id = id;
	}

}

子类


public class Reflect extends IdEntity {

	public String publicField;
	private String privateField;

	public Reflect(String publicField, String privateField) {
		super();
		this.publicField = publicField;
		this.privateField = privateField;
	}

	public String getPrivateField() {
		return privateField;
	}

	public void setPrivateField(String privateField) {
		this.privateField = privateField;
	}

}

测试类:

import java.lang.reflect.Field;

public class ReflectDemo {

	/**
	 * @Description: TODO
	 * @param @param args
	 * @return void
	 * @throws SecurityException
	 * @throws NoSuchFieldException
	 * @throws IllegalAccessException
	 * @throws IllegalArgumentException
	 * @throws
	 * @author ydecai
	 * @date 2019-1-23
	 */
	public static void main(String[] args) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
		Reflect re = new Reflect("zhangsan", "lisi");
		re.setId("111");
		//getDeclaredFiled获取本类属性
//		Field publicField = re.getClass().getDeclaredField("publicField");
//		System.out.println(publicField.get(re));
		Field publicField = re.getClass().getField("publicField");
		System.out.println(publicField.get(re));
		//getField获取公有属性,包括父类的,下面语句报错
		//Field privateField = re.getClass().getField("privateField");
		Field privateField = re.getClass().getDeclaredField("privateField");
		privateField.setAccessible(true);
		System.out.println(privateField.get(re));
		//获取父类私有属性并获取值
		Field fileId = re.getClass().getSuperclass().getDeclaredField("id");
		fileId.setAccessible(true);
		System.out.println(fileId.get(re));
	}

}

 

  • 31
    点赞
  • 89
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
Java反射机制是指在运行时动态地获取类的信息并操作类的属性、方法和构造方法,而无需在编译时确定这些信息。Java反射机制通过以下步骤来实现: 1. 获取Class对象:使用Class类中的静态方法获取类的Class对象,例如: ``` Class<?> clazz = Class.forName("com.example.MyClass"); ``` 2. 获取类的构造方法:使用Class类中的getConstructor()或getDeclaredConstructor()方法获取类的构造方法,例如: ``` Constructor<?> constructor = clazz.getConstructor(String.class, int.class); ``` 3. 创建类的实例:使用构造方法创建类的实例,例如: ``` Object obj = constructor.newInstance("example", 123); ``` 4. 获取类的属性:使用Class类中的getField()或getDeclaredField()方法获取类的属性,例如: ``` Field field = clazz.getField("name"); ``` 5. 修改类的属性:使用Field类中的set()方法修改类的属性,例如: ``` field.set(obj, "newName"); ``` 6. 获取类的方法:使用Class类中的getMethod()或getDeclaredMethod()方法获取类的方法,例如: ``` Method method = clazz.getMethod("doSomething", String.class); ``` 7. 调用类的方法:使用Method类中的invoke()方法调用类的方法,例如: ``` method.invoke(obj, "someArg"); ``` Java反射机制的应用场景包括框架开发、动态代理、注解处理器等。需要注意的是,反射机制虽然灵活,但由于是在运行时进行的,因此效率较低,而且容易出现类型不匹配、访问权限等问题,应谨慎使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值