反射的原理、及示例API ( java的特性之一)

反射: java的特性之一

1.        定义:反射机制是在运行时,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意个对象,都能 够调用它的任意一个方法。在java中,只要给定类的名字,就可以通过反射机制来获得类的所有信息。
这种动态获取的信息以及动态调用对象的方法的功能称为 Java 语言的反射机制

    描述: 是一种动态的程序加载和运行的机制
2.    原理: 反射的核心是JVM, jvm初次加载class文件时, 如果发现了未经编译的数据, 
         则将class文件返还编译器, 由编译器进行二次编译.
    反射的API:
        核心: Class, 类对象
3.            获取类对象的3种方法:
                1) 通过class关键字获取, 例: Class clazz = User.class;
                2) 通过getClass()获取, 例: User user = new User(); Class clazz = user.getClass();
                3) 通过全类名获取, 例: Class clazz = Class.forName("com.three.bean.User");
                    注: 检查异常, ClassNotFoundException
            通过Class对象获取实例: Object obj = clazz.newInstance();
        其它: Field, Method, Constructor, ...
    

4 、实现 Java 反射的类:
1 Class :表示正在运行的 Java 应用程序中的类和接口
注意: 所有获取对象的信息都需要 Class 类来实现。
2 Field :提供有关类和接口的属性信息,以及对它的动态访问权限。
3 Constructor :提供关于类的单个构造方法的信息以及它的访问权限
4 Method :提供类或接口中某个方法的信息
5 、反射机制的优缺点:
优点:
1 )能够运行时动态获取类的实例,提高灵活性;
2 )与动态编译结合
缺点:
1 )使用反射性能较低,需要解析字节码,将内存中的对象进行解析。
解决方案:
1 、通过 setAccessible(true) 关闭 JDK 的安全检查来提升反射速度;
2 、多次创建一个类的实例时,有缓存会快很多
3 ReflflectASM 工具类,通过字节码生成的方式加快反射速度
2 )相对不安全,破坏了封装性(因为通过反射可以获得私有方法和属性)

编程语言的分类: (是否可以在程序的运行阶段对程序内容进行改变)
    动态语言: 可以. 灵活但不稳定
    静态语言: 不可以. 稳定但缺乏灵活性

示例:

package com.three.test;

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import com.three.bean.User;

public class Test6 {

	public static void main(String[] args) {
		
		/* 获取类对象 */
		//类名.class
		Class clazz = User.class;
		
		//对象.getClass()
		User user = new User(1001, "韩梅梅", "6666");
		clazz = user.getClass();
		
		//Class.forName("全类名")
		try {
			clazz = Class.forName("com.three.bean.User");
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		
		/* 通过Class获取实例 */
		Object obj = null;
		try {
			obj = clazz.newInstance();
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		}
		
		/* 通过Class对象操作属性 */
		//操作单个属性
		try {
			//获取属性对象
//			Field userName = clazz.getField("userName"); //可以获取父类属性, 但有访问权限限制
			Field password = clazz.getDeclaredField("password"); //只可以获取本类属性, 但没有访问权限限制
			
			//设置访问权限
			password.setAccessible(true);
			
			//读取属性值
//			System.out.println(password.get(user));
			
			//为属性赋值
			password.set(obj, "9999");
//			System.out.println(password.get(obj));
			
		} catch (NoSuchFieldException e) {
			e.printStackTrace();
		} catch (SecurityException e) {
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		}
		
		//操作属性数组
		Field[] fields1 = clazz.getFields();
		Field[] fields2 = clazz.getDeclaredFields();
		
//		System.out.println("1:" + fields1.length);
//		System.out.println("2:" + fields2.length);
//		for(Field f : fields2) {
//			System.out.println("\t" + f.getName());
//		}
		
		/* 通过class对象操作方法 */
		//操作单个方法
		try {
			//获取指定名称和参数列表的方法
			Method m1 = clazz.getMethod("setUserName", String.class);
			Method m2 = clazz.getMethod("getUserName");
			
			//使用方法
//			System.out.println(m2.invoke(user));
			
//			System.out.println(m2.invoke(obj));
			m1.invoke(obj, "李雷");
//			System.out.println(m2.invoke(obj));
			
		} catch (NoSuchMethodException e) {
			e.printStackTrace();
		} catch (SecurityException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			e.printStackTrace();
		}
		
		//操作方法数组
		Method[] methods1 = clazz.getMethods();
		Method[] methods2 = clazz.getDeclaredMethods();
		
		System.out.println("1:" + methods1.length);
		for(Method m : methods1) {
			System.out.println("\t" + m.getName());
		}
		System.out.println("------------------------------------------");
		System.out.println("2:" + methods2.length);
		for(Method m : methods2) {
			System.out.println("\t" + m.getName());
		}
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值