美文网首页
JAVA反射

JAVA反射

作者: Jason_M_Ho | 来源:发表于2017-12-08 10:36 被阅读0次

    反射是Java 语言的特征之一,它允许运行中的Java程序获取自身的信息,并且可以操作类或对象的内部属性。通过反射,可以在运行时获得程序中每一个类型的成员和成员的信息。程序中一般的对象的类型都是在编译期就确定下来的,而Java反射机制可以动态地创建对象并调用其属性,这样的对象的类型在编译期是未知的。

    反射的核心是JVM在运行时才动态加载类或调用方法/访问属性,它不需要事先知道运行对象是谁。

    Java反射框架主要提供以下功能:

    1. 在运行时判断任意一个对象所属的类;
    2. 在运行时构造任意一个类的对象;
    3. 在运行时判断任意一个类所具有的成员变量和方法;
    4. 在运行时调用任意一个对象的方法

    当我们在使用IDE(如Eclipse,IDEA)时,当我们输入一个对象或类并想调用它的属性或方法时,一按点号,编译器就会自动列出它的属性或方法,这里就会用到反射。

    反射最重要的用途就是开发各种通用框架。

    操作反射常用的类:Constructor、Field、Method、Array位于java.lang.reflect包中。
    通过反射获取类的信息分为两步:

    1. 获取Class对象

      • 调用对象的getClass()方法
      • 调用类的class属性
      • 调用Class类的forName()方法
    2. 通过Class对象获取信息

      • 访问Class对应的类的构造方法
      • 访问Class对应的类的方法
      • 访问Class对应的类的属性

    访问Class对应的类的构造方法

    访问Class对应的类的构造方法 说明
    Constructor getConstructor(Class[ ] params) 返回此Class对象所包含的类的指定的public构造方法
    Constructor[ ] getConstructors( ) 返回此Class对象所包含的类的所有的public构造方法
    Constructor getDeclaredConstructor(Class[ ] params) 返回此Class对象所包含的类的声明的指定的构造方法
    Constructor[ ] getDeclaredConstructors( ) 返回此Class对象所包含的类的声明的所有构造方法

    访问Class对应的类的方法

    访问Class对应的类的方法 说明
    Method getMethod(String name, Class[ ] params) 返回此Class对象所包含的类的指定的public方法
    Method[ ] getMethods( ) 返回此Class对象所包含的类的所有的public方法
    Method getDeclaredMethod(String name, Class[ ] params) 返回此Class对象所包含的类的声明的指定的方法
    Method[ ] getDeclaredMethods( ) 返回此Class对象所包含的类的声明的所有方法

    访问Class对应的类的属性

    访问Class对应的类的属性 说明
    Field getField(String name) 返回此Class对象所包含的类的指定的public属性
    Field[ ] getFields( ) 返回此Class对象所包含的类的所有的public属性
    Field getDeclaredField(String name) 返回此Class对象所包含的类的声明的指定的属性
    Field[ ] getDeclaredFields( ) 返回此Class对象所包含的类的声明的所有属性

    Array类操作数组

    Array类里定义了大量静态方法,通过类名直接调用。
    这里的Array和集合框架里的Arrays不一样,他们的相同点都是定义了大量静态方法,都是数组的操作,Array位于java.lang.relect包,是动态创建并操作数组;Arrays位于java.util包,是一个工具类,其中包含了一些方法可直接实现数组的排序、搜索等。

    下面结合代码做下说明:

    public class Demo {
              //调用class对象会抛出大量异常,这里为简化代码,直接抛给jvm
        public static void main(String[] args) throws Exception {
            Student stu = new Student("张三",23);
            
            //获取Class对象
            //方法一:forName()方法,必须要完整的包名加类名
            Class<?> stuClass = Class.forName("reflect.Student");
            //方法二:类的class属性
            Class<?> cla1 = Student.class;
            //方法三:getClass()方法
            Class<?> cla2 = stu.getClass();
            System.out.println(stuClass.getName());
            
            //获取声明的所有构造方法
            Constructor<?>[] constructor = stuClass.getConstructors();
            for (int i = 0; i < constructor.length; i++) {
                System.out.println(constructor[i].toString());
            }
            //获取指定的一个有参构造方法
            Constructor<?> con1 = stuClass.getDeclaredConstructor(String.class,int.class);
            Object obj = con1.newInstance("tom",20);  //使用Class对象的newInstance()方法创建对象
            ((Student)obj).shwoInfo();
            
            //获取声明的所有属性
            Field[] fields = stuClass.getDeclaredFields();
            for (int i = 0; i < fields.length; i++) {
                System.out.println(fields[i].toString());
            }
            //获取public name属性,并改变其值
            Field fieldname = stuClass.getDeclaredField("name");
            fieldname.set(stu, "王小二");//set方法给属性赋值
            String nameStr = (String) fieldname.get(stu); //get方法取出属性的值
            System.out.println(nameStr);
            //获取private age属性,并改变其值
            Field fieldAge = stuClass.getDeclaredField("age");
            fieldAge.setAccessible(true); //改变属性访问权限
            fieldAge.set(stu, 58);
            int ageInt = fieldAge.getInt(stu);
            System.out.println(ageInt);
            
            //获得声明的所有方法
            Method[] methods = stuClass.getDeclaredMethods();
            for (int i = 0; i < methods.length; i++) {
                System.out.println(methods[i].toString());//以字符串形式打印出方法名
            }
            //获得声明的指定方法,传值并调用
            Method m1 = stuClass.getDeclaredMethod("sayHelloTo", String.class);
            m1.invoke(null, "jerry");//这里的null是因为调用的方法是静态方法,不用传对象
            //获得其中一个方法,直接调用
            Method m2 = stuClass.getDeclaredMethod("shwoInfo");
            m2.invoke(stu);//用invoke()调用方法时需要传一个对象
    
            //Array操作数组
            Object arr = Array.newInstance(int.class, 3);//创建元素为3的int数组
            for (int i = 0; i < Array.getLength(arr); i++) {
                Array.set(arr, i, 2*i); //循环给元素赋值
            }
            for (int i = 0; i < Array.getLength(arr); i++) {
                System.out.println(Array.get(arr, i));//循环取值
            }
            Array.set(array, 0, 99);//指定元素赋值
            System.out.println(Array.get(array, 0));//指定元素取值
        }
    }
    

    相关文章

      网友评论

          本文标题:JAVA反射

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