反射

作者: 月寒兮夜凉 | 来源:发表于2016-11-04 18:50 被阅读48次

优秀文章

博客:Rittel_Liu > Java反射的作用
博客园:Rollen Holt > Java反射详解


说明

  • Class类是反射机制的基础;
  • 在Java中,所有的类都直接或者间接继承了Object这个类,而在Object类中定义了一个getClass()方法用于返回一个类型为Class的对象,所以Java的所有类都有一个相应的Class对象;
  • Class对象用于表示当前类的类型信息,当我们编写一个类,编译完成后,在生成的.class文件中,就会产生这个Class对象,对象中存放的都是当前操作类的类型信息;
  • 程序运行时,Java虚拟机(JVM)会检查操作类的Class对象是否已被加载。如果没有被加载,JVM会根据类的名称找到.class文件并加载它。一旦某个类型的Class对象已被加载到内存,就可以用它来产生该类型的所有对象,故所有类的对象都是Class类的实例
  • Class类没有公共构造方法;
  • 反射是关于对象对自身成员的描述;
  • 反射机制主要体现在Class类对操作类成员的封装,通过对Class类实例的操作达到做操作类的控制;

Class类的获取

  1. 使用Object类的getClass()方法;
  2. 使用Class类中的静态forName("类名/接口名")方法;
  3. 使用.class直接获得调用类的Class对象;

Java反射

[PS:下文Demo为演示类,demo为Demo的对象]

  1. 获取Class实例
  • 通过对象获取实例:
    • Class<?> clazz = demo.getClass();
  • 通过静态方法获取实例:
    • Class<?> clazz = Class.forName("com.ais.Demo");// 参数类名必须包含各级包名
  1. 利用Class实例实现反射操作
  • 获取演示类的新对象:
    • Demo newDemo = (Demo) clazz.newInstance();
  • 获取演示类的所有成员属性:
    • Field[] field = clazz.getDeclaredFields();
  • 获取演示类的所有成员方法:
    • Method[] method = clazz.getDeclaredMethods();
  • 获取演示类的所有成员方法可抛出的所有异常:
    • Class<?>[] method = method.getExceptionType();
  • 获取演示类的父类Class实例:
    • Class<?> superClazz = clazz.getSuperclass();
  • 获取演示类的所有实现接口Class实例:
    • Class<?>[] interfaceClazzs = clazz.getInterfaces();
  • 获取Class实例的访问修饰符:
    • int modifiers = clazz.getModifiers();// 获取修饰符的整数编码
    • String modifiersStr = Modifier.toString(modifiers);// 解码成可读字符串
  • 获取类或其成员上注解:
    • [clazz | field | method].isAnnotationPresent(注解类名.class)// 验证注解是否存在
    • [clazz | field | method].getAnnotation(注解类名.class)// 获取注解

实例

>编写类解析器
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

public class ClassParser {
    /**
     * @description print class info
     */
    public static void print(String type, Class<?> clazz){

        System.out.println("************");
        
        // 获取解析对象名称
        String name = clazz.getName().toString();
        String[] names = name.split("\\.");
        System.out.println(type+":"+names[names.length-1]);

        System.out.println("访问修饰:"+Modifier.toString(clazz.getModifiers()));
        System.out.println("完整名称:"+name);
        
        // 解析成员属性
        Field[] fields = clazz.getDeclaredFields();
        for(Field field : fields){
            System.out.println("成员属性:"+Modifier.toString(field.getModifiers())+" "+field.getType().getName()+" "+field.getName());
        }
        
        // 解析成员方法
        Method[] methods = clazz.getDeclaredMethods();
        for(Method method : methods){
            // 获取方法参数
            Class<?>[] parameters = method.getParameterTypes(); 
            String parametersStr = "";
            for(Class c : parameters){
                parametersStr = parametersStr+c.getName()+", ";
            }
            if(parametersStr.length()!=0){
                parametersStr = parametersStr.substring(0, parametersStr.length()-2);
            }
            // 获取方法异常
            Class<?>[] exceptions = method.getExceptionTypes(); 
            String exceptionStr = "";
            if(exceptions!=null && exceptions.length>0){
                exceptionStr = " throws ";
                for(Class c : exceptions){
                    exceptionStr = exceptionStr+c.getName()+", ";
                }
                if(exceptionStr.length()!=5){
                    exceptionStr = exceptionStr.substring(0, exceptionStr.length()-2);
                }
            }
            System.out.println("成员方法:"+Modifier.toString(method.getModifiers())+" "+method.getReturnType().getName()+" "+method.getName()+"("+parametersStr+")"+exceptionStr);
        }
        
        // 解析接口
        Class<?>[] interfaceClazzs = clazz.getInterfaces();
        for(Class cls : interfaceClazzs){
            print(type+"的接口", cls);
        }
        
        // 解析父类
        Class<?> superClazz = clazz.getSuperclass();
        if(superClazz!=null){
            print(type+"的父类", superClazz);
        }
    }
    
}
>执行
    public static void main(String[] args) {
        try {
            Class<?> clazz = Class.forName("org.apache.poi.hssf.usermodel.HSSFCellStyle");
            ClassParser.print("演示类", clazz);    
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } 
    }
>结果

演示类:Demo
访问修饰:public
完整名称:com.ais.test.Demo
成员属性:public int num
成员属性:public java.lang.String str
成员方法:public void interfaceFunc()
成员方法:public void func(int, java.lang.String) throws java.lang.Exception


演示类的接口:DemoInterface
访问修饰:public abstract interface
完整名称:com.ais.test.DemoInterface
成员方法:public abstract void interfaceFunc()


演示类的父类:SuperDemo
访问修饰:public
完整名称:com.ais.test.SuperDemo
成员属性:public int number
成员方法:private java.lang.String superFunc()


演示类的父类的父类:Object
访问修饰:public
完整名称:java.lang.Object
成员方法:protected void finalize() throws java.lang.Throwable
成员方法:public final void wait(long, int) throws java.lang.InterruptedException
成员方法:public final native void wait(long) throws java.lang.InterruptedException
成员方法:public final void wait() throws java.lang.InterruptedException
成员方法:public boolean equals(java.lang.Object)
成员方法:public java.lang.String toString()
成员方法:public native int hashCode()
成员方法:public final native java.lang.Class getClass()
成员方法:protected native java.lang.Object clone() throws java.lang.CloneNotSupportedException
成员方法:private static native void registerNatives()
成员方法:public final native void notify()
成员方法:public final native void notifyAll()


最后

  • 初探反射;

相关文章

  • Java基础之反射

    Java基础之反射 反射基本介绍 反射的使用通过反射调用属性和方法通过反射获取配置文件 反射基本介绍 Java反射...

  • 镜面反射矩阵的推导

    镜面反射是以镜面作为反射平面,实物与反射物到反射平面的距离相等,实物与反射物方向相反,所以,反射矩阵由反射平面确定...

  • reflect.go包学习_之二 指针操作提高反射性能 反射应用

    reflect.go包学习_之二 指针操作提高反射性能 反射应用 反射创建实例 反射信息、反射调用方法、反射修改值...

  • Java互联网公司-经典面试题附答案

    基础:Java 反射?反射有什么缺点?你是怎么理解反射的(为什么框架需要反射)?优点:反射具有解耦性,缺点:反射属...

  • Java反射与joor反射库的使用

    java原生反射的使用 反射构造对象 反射方法 反射字段 joor反射库的使用 github:https://gi...

  • Java反射

    什么是反射? 反射的作用? 反射性能优化?

  • 反射三定律

    反射第一定律:反射可以将interface类型变量转换成反射对象 反射第二定律:反射可以将反射对象还原成inter...

  • 反射

    1.反射是什么?反射的定义,概念 2.为什么要学反射? 3.怎么使用反射?反射的使用

  • 一周岁前做好两件事,孩子就不会语言迟缓,保证口齿伶俐

    与语言发展相关的原始反射有四个:张口反射、足跖反射、抓握反射和手拉反射,每个反射的发生、发展和整合都是次第进行的,...

  • 面试官问go反射第一弹

    目录 反射概念 reflect包 反射类型(Type)和种类(Kind) 反射类型(Type)使用 反射类型对象(...

网友评论

    本文标题:反射

    本文链接:https://www.haomeiwen.com/subject/bcniuttx.html