Java反射机制的知识点有哪些

类的声明周期

java源代码----->javac-------------->java字节码文件-------------->java----------------->类对象(所在内存空间:元空间,本地内存)------------------------new--------->实例化对象------------------gc------------->卸载对象

不同阶段都可以获取类对象

  • 对象.getClass() (内存阶段)

  • Test.class (元空间)

  • class.forName(“类全名:包名+类名”) :(硬盘)都没有进入内存空间就可以拿到对象

举例来说,当我们使用jdbc操作数据库时,在这个类进入内存之前,我们可以使用该类的完整名称——包名+类名,调用该类并使用

获取Class类对象的方式的场景

  • Class.forName(“类全名”) :多用于配置文件,将类名定义在配置文件中,读取配置文件,加载类

  • 类名.class : 多用于参数的传递

  • 对象名.getClass():多用于对象获取类对象

总结:同一个类加载器加载的文件在一次程序运行过程中,只会被加载一次,无论使用哪种方式获得的类对象都是同一个

代码示例:

package com.reflect;
public class TestReflectPerson {
    public static void main(String[] args) throws ClassNotFoundException {
        //1.class.forName()
        Class class1=Class.forName("com.reflect.Person");
        System.out.println(class1);
        //2.类名.class
        Class class2=Person.class;
        System.out.println(class2);
        //2.对象名.getClass()
        Class class3=new Person().getClass();
        System.out.println(class3);
        System.out.println(class1==class2);  //true
        System.out.println(class2==class3);  //true
    }
}

class类对象的功能

获取成员变量 : 取所有:类对象.getDeclaredFields() ,获取一个:类对象.getDeclaredField()

  • 设置值 set(Object obj,Object value)

  • 获取值 get(Object obj)

获取任意权限修饰的成员变量获取设置值,需要使用setAccessible(true)-----暴力反射

成员方法: 类对象.getDeclaredMethods()

执行方法 invoke(Object object ,Object… agrs) (参数个数任意,可有可无)

获取方法名getName()

构造方法: 类对象.getDeclaredConstructors()

虽然必须具备无参构造方法,但是使用 newInstance() 方法可以省略获取构造方法并获取对象的步骤

该方法需要实际情况构造方法赋实参

//获得构造方法对象,
        Constructor cons1 = pcla.getDeclaredConstructor(String.class, int.class);
        Person p2 = (Person)cons1.newInstance("李四",19);
        System.out.println("p2:"+p2.getName());

newInstance()如果是创建无参构造方法去创建对象,可以使用类对象来创建对象,跳过获得构造方法对象

获取

获得类的名称:getName() 打印出全名:类名+包名

只想打印单独类名:getSimpleName()

获取类的成员变量名称

属性文件:内容以等号连接形如k=v,

代码示例:

package com.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class TestReflectPerson {
    public static void main(String[] args) throws Exception {
       /* //1.class.forName()
        Class class1=Class.forName("com.reflect.Person");
        System.out.println(class1);
        //2.类名.class
        Class class2=Person.class;
        System.out.println(class2);
        //2.类名.getClass()
        Class class3=new Person().getClass();
        System.out.println(class3);
        System.out.println(class1==class2);
        System.out.println(class2==class3);*/
        //获取对象
        Class tclass=Class.forName("com.reflect.Person");
        //通过类对象获取成员变量们
        Field[] fields = tclass.getDeclaredFields();
        System.out.println("获取Person对象的所有属性对象");
        for (Field field:fields){
           System.out.println(field);
       }
        //指定获取Person对象的属性对象
        System.out.println("指定获取Person对象的属性对象");
        Field age=tclass.getDeclaredField("age");
        System.out.println("age:"+age);
        //通过类对象获取所有的构造方法
        Constructor[] constructors = tclass.getDeclaredConstructors();
        System.out.println("获取Person的所有构造方法对象");
        for (Constructor constructor:constructors){
            System.out.println(constructor);
        }
        //通过类对象获取无参的构造方法
        Constructor constructor = tclass.getDeclaredConstructor();
        System.out.println("constructor:"+constructor);
        //通过类对象获取有参的构造方法
        Constructor constructor1 = tclass.getDeclaredConstructor(String.class,int.class);
        System.out.println("constructor1:"+constructor1);
        //通过类对象获取所有的成员方法
        Method[] methods = tclass.getDeclaredMethods();
        for (Method method:methods){
            System.out.println("method:"+method);
        }
        //通过类对象获取getAge成员方法
        Method getAge = tclass.getDeclaredMethod("getAge");
        System.out.println("getAge:"+getAge);
        //通过类对象获取getAge成员方法
        Method setAge = tclass.getDeclaredMethod("setAge", int.class);
        System.out.println("setAge:"+setAge);
    }
}

获取成员变量代码示例:

package com.reflect;
import java.lang.reflect.Field;
public class TestField {
    public static void main(String[] args) throws Exception {
        Class pcla=Person.class;
        /*//获取公共访问权限的成员变量
        Field[] fields = pcla.getFields();
        for (Field field:fields){
            System.out.println("getFild:"+field);
        }
        System.out.println();
        //获取所有访问权限的成员变量
        Field[] fielddes = pcla.getDeclaredFields();
        for (Field field:fielddes){
            System.out.println("field:"+field);
        }*/
        Field name = pcla.getDeclaredField("name");
        System.out.println(name);
        Person person=new Person();
        //暴力反射:获取任意访问权限修饰符的安全检查
        name.setAccessible(true);
        //获取公共成员变量的值
        Object value = name.get(person);
        System.out.println(value);
        //获取任意访问权限的成员变量的值
        Object value2 = name.get(person);
        System.out.println("value2:"+value2);
        //设置任意访问权限的成员变量的值
        name.set(person,"张三");
        Object value3=name.get(person);
        System.out.println("name:"+value3);
    }
}

如何获取私有变量的值

//暴力反射:获取任意访问权限修饰符的安全检查
name.setAccessible(true);

根据有无主方法判断进程和线程

进程:含有自己的主方法,可以依托自己的主方法启动,叫做进程

线程:没有自己的主方法,需要依赖其他工具来运行

例如:servlet就需要借助tomcate来运行,tomcate有自己的一个主方法

反射出现的背景(记住)

举例:在servlet通过借助工具tomcate来运行时,tomacate运行项目时访问不到类的资源,由此产生了反射

tomcate为什么拿不到new的对象

详解:tomcate不可能通过new的方式来调用,因为tomacate是先产生写好的,类是后来写的,所以tomcate不知道new的对象的是什么,可以通过包扫描的方式来获取文件路径,但是这样也无法使用new的方式,由此产生了反射。

ate有自己的一个主方法

反射出现的背景

举例:在servlet通过借助工具tomcate来运行时,tomacate运行项目时访问不到类的资源,由此产生了反射

tomcate为什么拿不到new的对象?

详解:tomcate不可能通过new的方式来调用,因为tomacate是先产生写好的,类是后来写的,所以tomcate不知道new的对象的是什么,可以通过包扫描的方式来获取文件路径,但是这样也无法使用new的方式,由此产生了反射。

tomcate想调用doGet,doPost的方法时,因为这两个方法不是静态的,必须通过new对象才能调用,但是tomcate又不能创建对象,所以由此产生反射来获取文件

以上就是Java反射机制的知识点有哪些的详细内容,更多请关注其它相关文章!