• 细说java系列之反射


    什么是反射

    反射机制允许在Java代码中获取被JVM加载的类信息,如:成员变量,方法,构造函数等。
    在Java包java.lang.reflect下提供了获取类和对象反射信息的相关工具类和接口,如:Field,Method,Constructor等。

    使用反射可以做什么事情

    反射通常被用于需要检查或修改应用程序运行时行为的编程中,它是一个非常有用的技术。
    具体来讲,可以在如下场景中使用反射机制:

    • 功能扩展,应用程序可以通过反射创建一个具备完整限定名的类实例,从而使用一个外部的用户自定义的类。
    • 在可视化的开发环境中浏览类信息,其实在eclipse中通过快捷键Ctrl + O显示的类信息就是利用反射机制实现的。
    • 用于程序调试器和测试工具

    反射的缺点

    虽然反射机制可以增强应用程序的功能和使用场景,但并非在任何情况下都适合使用的,因为反射机制本身存在一些固有的缺点。

    1. 性能损耗,反射需要动态解析类型,相比起不使用反射的情况是存在性能损耗的,所以在性能比较敏感或重要的应用程序中不要使用反射。
    2. 安全限制,反射需要的运行时权限在安全管理器(SecurityManager)下是被禁止的,比Applet程序中。
    3. 类结构被暴露,由于反射允许在非反射代码中执行一些非法操作,打破了Java原本的抽象模型,可能会影响到平台的行为与升级。

    应用实践

    /**
     * 利用反射机制获取被JVM加载的类信息,实例化类对象。
     * @desc org.chench.test.java.UserReflector
     * @author chench9@lenovo.com
     * @date 2017年11月30日
     */
    public class UserReflector {
    	public static void main(String[] args) throws Exception {
    		String className = "org.chench.test.java.User";
    		
    		// 通过类的完整限定名获取其Class对象
    		Class<?> userClass = Class.forName(className);
    		
    		// 使用反射机制获取类的构造函数列表
    		Constructor[] constructorArr =  userClass.getConstructors();
    		for(Constructor constructor : constructorArr) {
    			// 构造函数的名称
    			String name = constructor.getName();
    			System.out.println("constructor name: " + name);
    			
    			// 构造函数的参数个数
    			int count = constructor.getParameterCount();
    			System.out.println("constructor parameter count: " + count);
    			
    			// 构造函数的参数列表
    			Parameter[] parameters = constructor.getParameters();
    			for(Parameter parameter : parameters) {
    				// 获取参数类型
    				Class<?> parameterType = parameter.getType();
    				// 获取参数名
    				String parameterName = parameter.getName();
    				System.out.println(parameterName + " Type: " + parameterType);
    			}
    			
    			// 通过构造函数实例化类对象
    			if(count <= 0) {
    				User user =	(User) constructor.newInstance();
    				System.out.println("user instance: " + user);
    			}else if(count == 1) {
    				User user = (User) constructor.newInstance(new Object[] {"Zhang San"});
    				System.out.println("user instance: " + user);
    			}else if(count == 2) {
    				User user = (User) constructor.newInstance(new Object[] {"Li Si", 26});
    				System.out.println("user instance: " + user);
    			}
    			System.out.println("----------");
    		}
    		
    		// 使用反射机制获取类的成员变量
    		Field[] fields = userClass.getDeclaredFields();
    		for(Field field : fields) {
    			// 变量类型
    			Class<?> fieldType = field.getType();
    			// 变量名称
    			String fieldName =	field.getName();
    			// 变量修饰符,private:2,public:1, protected: 4,默认类型: 0
    			int fieldModifier = field.getModifiers();
    			System.out.println("field info, name: " + fieldName + ", type: " + fieldType + ", modifier: " + fieldModifier);
    		}
    		
    		// 使用反射机制获取类的方法
    		Method[] methods = userClass.getDeclaredMethods();
    		for(Method method : methods) {
    			// 方法返回值类型
    			Class<?> returnType = method.getReturnType();
    			// 方法名
    			String methodName = method.getName();
    			// 方法参数个数
    			int count = method.getParameterCount();
    			System.out.println("method info, name: " + methodName + ", return type: " + returnType + ", parameter count: " +count);
    		}
    	}
    }
    

    实际上,在应用编程中使用反射机制最多的场景主要是如下2个方面:

    1. 在注解解析器中通过反射获取类,方法或成员变量的注解信息。
    2. 在动态代理类中使用反射机制调用方法执行。

    【参考】
    https://docs.oracle.com/javase/tutorial/reflect/TOC.html

  • 相关阅读:
    pthread线程内存布局
    用户空间实现线程 内核实现线程 线程的调度
    堆 虚拟内存
    Operating System-Thread(3)用户空间和内核空间实现线程
    Linux进程地址空间与虚拟内存
    虚拟地址
    物理内存,虚拟内存,进程地址空间
    CPU中MMU的作用
    进程地址空间与虚拟存储空间的理解
    虚拟地址空间
  • 原文地址:https://www.cnblogs.com/nuccch/p/7928336.html
Copyright © 2020-2023  润新知