反射

反射

一、概述

    反射其实就是动态加载一个指定的类,并获取该类中的所有的内容。而且将字节码文件封装成对象,并将字节码文件中的内容都封装成对象,这样便于操作这些成员。简单说:反射技术就是对一个类进行解剖,动态获取对象信息以及动态调用对象的方法的功能。

    要想要对字节码文件进行解剖,必须要有字节码文件对象。如何获取字节码文件对象呢?

通过Class类就可以获得字节码文件对象。Class类是描述Java类这类事物的一个类,比如类的名字,访问属性,字段名称,方法名称等等。

反射的基本步骤是什么?

1.获得Classd对象,就是获取到指定的名称的字节码文件对象。

2.实例化对象,获得类的属性、方法或构造函数。

3.访问属性、方法、调用构造函数创建对象。


二、反射的用法

1.获取Class对象的三种方式

第一种:通过每个对象都具备的方法getClass来获取

好处:如果拿到了对象,不知道是什么类型 ,可用于获得对象的类型

弊端:必须要明确具体的类并且要创建该类对象,才可以调用getClass方法。不利于程序的扩展。

如:

Object obj = new Person();//包中有一个Person对象

Class clazz1 = obj.getClass();// 获得对象具体的类型

第二种:根据任何数据都具备的静态的属性.class来获取其对应的Class对象。

好处:如果是明确地获得某个类的Class对象  主要用于传参。

弊端:相对简单,但还是必须先明确该类,同样不利于程序的扩展。

如:

Class clazz2 = Person.class;  

第三种:使用Class类中的静态方法forName。

好处:根据给定的类名来获得,用于加载。扩展性最强。

String classname ="cn.itcast.reflect.Person";// 来自配置文件

Class clazz = Class.forName(classname);// 此对象代表Person.class

2.构造函数

获取了字节码文件对象后,最终都需要创建指定类的对象。

创建对象的两种方式(其实就是对象在进行实例化时的初始化方式)

1)调用空参数的构造函数:使用了Class类中的newInstance()方法。

2)调用带参数的构造函数:先要获取指定参数列表的构造函数对象,然后通过该构造函数的对象的newInstance(实际参数)进行对象的初始化。


综上所述,第二种方式,必须要先明确具体的构造函数的参数类型,不便于扩展。

所以一般情况下,被反射的类,内部通常都会提供一个公有的空参数的构造函数。

1)获取空参数的构造函数

<span style="font-size:14px;">//加载类获取字节码文件对象

   Classclazz = Class.forName("cn.itcast.bean.Person");

// 调用空参数构造函数实例化对象

   Object obj = clazz.newInstance();
</span>

2)获取带参数的构造函数

<span style="font-size:14px;">public static void getConstuor() throws Exception {
	Class clazz = Class.forName("cn.itcast.bean.Person");
	//既然类中没有空参数的构造函数,那么只有获取指定参数的构造函数,用该函数来进行实例化。
	//获取一个带参数的构造器。
	Constructor constructor = clazz.getConstructor(String.class,int.class);
	//想要对对象进行初始化,使用构造器的方法newInstance();
	Object obj = constructor.newInstance("zhagnsan",30);
	//获取所有构造器。
	Constructor[] constructors = clazz.getConstructors();//只包含公共的
	constructors = clazz.getDeclaredConstructors();//包含私有的
	for(Constructor con : constructors) {
		System.out.println(con);
	}
}</span>

3.字段

getDeclaredField("age"):获取一个类中所有方法,包含私有。           

setAccessible(true); 对私有字段的访问取消权限检查,即暴力反射。


<span style="font-size:14px;">//获取字节码文件中的字段。
	public static void getFieldDemo() throws Exception {
		
		Class clazz = Class.forName("cn.itcast.bean.Person");
		
		Field field = null;//clazz.getField("age");//只能获取公有的,
		
		field = clazz.getDeclaredField("age");//只获取本类,但包含私有。 
		
		//对私有字段的访问取消权限检查。
		field.setAccessible(true);
		//初始化对象
		Object obj = clazz.newInstance();
		
		field.set(obj, 89);//set方法设置变量值	
		
		Object age = field.get(obj);//get方法获得变量值
		
		System.out.println(age);
		
	}</span>

4.方法

使用getMethod()方法可以获得方法,并使用invoke()来指定方法。

<span style="font-size:14px;">public static void method_2() throws Exception {
	Class clazz = Class.forName("cn.itcast.bean.Person");
	//获取指定名称的方法。
	Method method = clazz.getMethod("show", int.class,String.class);
	//想要运行指定方法,当然是方法对象最清楚,为了让方法运行,调用方法对象的invoke方法即可,但是方法运行必须要明确所属的对象和具体的实际参数。
	Object obj = clazz.newInstance();
	method.invoke(obj, 39,"hehehe");//执行一个方法
}</span>

与访问类中的私有字段一样,访问类中的私有方法时也要使用暴力反射。

<span style="font-size:14px;">public static void method_3() throws Exception {
	Class clazz = Class.forName("cn.itcast.bean.Person");
	//想要获取私有方法。必须用getDeclearMethod();
	Method method = clazz.getDeclaredMethod("method", null);
	// 私有方法不能直接访问,因为权限不够。非要访问,可以通过暴力的方式。
	method.setAccessible(true);//一般很少用,因为私有就是隐藏起来,所以尽量不要访问。
}</span>

要想访问类中的静态方法则,invoke方法的第一个参数为null。如下:

<span style="font-size:14px;">public static void method_4() throws Exception {
	Class clazz = Class.forName("cn.itcast.bean.Person");
	Method method = clazz.getMethod("function",null);
	method.invoke(null,null);//第一个null代表使用的是静态方法,第二个null是函数没有参数
}</span>



三、数组的反射


数组的反射具有以下特点:

1.具有相同维数和类型的数组具有相同的Class对象

2.基本类型的一维数组可以当作Object类型使用,不能当作Object[]类型使用。

非基本类型的一维数组既可以当作Object类型使用,也可以当作Object[]类型使用。

下面看一个例子:

<span style="font-size:14px;">//数组的反射,如果是基本类型,可以打印出
	private static void printObj(Object obj) { //因为一维数组可以当作Object类型使用,这里可以用Object
		//判断是否是数组
		if(obj.getClass().isArray()){
			int len = Array.getLength(obj);//获得长度
			for (int i = 0; i < len; i++) {
				System.out.println(Array.get(obj, i));//获得数组对象的第i个元素
			}
		}else{
			System.out.println(obj);
		}
		
	}</span>

主函数:

<span style="font-size:14px;">public static void main(String[] args) {
		//int[] a = {1,6,5,12};
		//printObj(a);//是数组的话,可以打印出数组的内容
		Ball b = new Ball();
		printObj(b);//不是数组对象,就直接打印出该对象即可

	}</span>

综上所述,可以看出反射技术的优势,它在框架的使用中很频繁,是一门必须掌握的技术。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值