Java反射机制

很早就听说了大名鼎鼎的反射机制,最近终于把它学了,总结一下Java反射机制的学习。

 

1.获得Class类。在JVM中,加载一个类的时候,会在堆中生成该类对应的Class类,通过Class类可以获得该类的所有属性和方法,Class类就像镜子一样反射着该类的所有内容。所以,在使用Java反射机制的时候,第一步就是获得该类的Class类(Class类位于java.lang包下面),获得Class类有以下几种方法:

1)Class.forName(类的全限定名);如:Class<?> clazz = Class.forName("java.lang.String");

2) 通过getClass()方法;getClass是Object类的方法,通过类的任一实例调用getClass()。

3)运用getSuperClass(),获得父类的Class类。

4).class语法:Java的八种基本类型,类名,数组,clazz = String.class;clazz = int.class;clazz = int[].class;

5) .TYPE语法:八种基本类型的包装类型和Void,clazz = Integer.TYPE;clazz = Void.TYPE;

 

2.通过反射机制获得该类的实例。得到Class类之后,有以下方法可以获得该类的实例,假设D类。

  1)运行Class类的newInstance()方法,相当于调用无参构造方法:D d = clazz.newInstance();

  2)运用getDeclaredConstructor()方法。由于newInstance(方法只能调用无参构造方法,局限性太大。

        public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)

   该方法的参数是构造方法参数类型所对应的Class类,假设D类有构造方法:public D(String name,int age){}。那么获得该构造方法所对应的类:

       Constructor cst = clazz.getDeclaredConstructor(new Class[]{String.class,int.class});

     Constructor类有一个实例对象的方法newInstance(Object...params),参数列表和构造方法对应

     D d = (D) cst.newInstance("cxy",25);//或者cst.newInstance(new Object[]{"cxy",25});

3.通过Java反射机制完成对类的私有成员变量的访问和修改

类中的成员变量对应的类是Field,方法对应的类是Method,通过它们可以完成对成员变量和方法的操作。下面演示如何通过Java的反射机制完成对类的私有成员变量的访问和修改(这个对我的冲击挺大的),解释都在代码中了:

package com.ssy.reflection;

import java.lang.reflect.Field;

/*
 * 该类演示了如果通过java的反射机制获得和修改类的私有变量
 */
public class ChangePrivate {

	/**
	 * 通过Class获得成员变量name所对应的Field类,Field的父类是AccessibleObject,它有一个方法,setAccessible(boolean flag)
	 * 如果flag=true,就取消访问控制检查,这样即使name是私有的,我们也能去访问它
	 * 注意
	 */
	public static void main(String[] args)throws Exception {
		PrivateT pt = new PrivateT();
		//获得对应的Class类
		Class<?> clazz = pt.getClass();
		//获得name字段对应的类
		Field field = clazz.getDeclaredField("name");
        field.setAccessible(true);//取消访问控制检查
        //获得name的值,get()的参数表示获取哪个实例的值
        String name = (String) field.get(pt);
        System.out.println(name);
        //修改name的值,set(Object obj,Object value),obj表示修改哪个实例的name,value表示修改的值
        field.set(pt, "xlf");
        System.out.println(pt.getName());
	}

}

class PrivateT {
	private String name = "cxy";
	
	public String getName(){
		return this.name;
	}
}

 最后在说一句:反射是的前提是运行时,不是编译时,也就说以上的所有全部是在运行时完成的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值