java反射机制

目录

1.1、概述

1.2、类加载器

1.3、加载配置文件

1.3、所有类型的Class对象

2、得到Class的几种方式 

3.通过class对象 获取一个类的构造方法

3.1Constructor 创建对象

3.2获取一个类的构造方法

4、获取Method

4.1、通过class对象 获取一个类的方法

4.2、Method 执行方法

案例

5、获取Field

5.1、通过class对象 获取一个类的属性

5.2、Field 属性的对象类型 

案例



1.1、概述

   java的反射是指程序在运行期可以拿到一个对象的所有信息,这种动态获取信息以及动态调用对象的功能称为反射。比如:你的代码2022年上线,2023年你不想代码关机,想代码继续使用且你可以在使用情况下对代码进行增加、修改等操作时就要使用反射。

1.2、类加载器

    Java类加载器是Java运行时环境的一部分, 负责动态加载Java类到Java虚拟机的内存空间中。

    java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、App ClassLoader。

     BootstrapClassLoader(引导启动类加载器): 嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib(就是JDK所安装下的目录)下的类库,引 导启动类加载器无法被应用程序直接使用。

    ExtensionClassLoader(扩展类加载器): ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。 是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类 库。 它的父加载器是BootstrapClassLoader App   。

   ClassLoader(应用类加载器):App ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文 件。我们平时使用加载一个配置文件或加载一个类使用的都是应用类加载器。 它的父加载器为Ext ClassLoader。

 执行流程:双亲委派模型:如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求 转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的 启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类) 时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。

1.3、加载配置文件

给项目添加resource root目录

       如图

通过类加载器加载资源文件

     默认加载的是src路径下的文件,但是当项目存在resource root目录时,就变为了加载 resource root下的文件了。

1.3、所有类型的Class对象

   要想了解一个类,必须先要获取到该类的字节码文件对象. 在Java中,每一个字节码文件,被夹在到内存后,都存在一个对应的Class类型的对象。

2、得到Class的几种方式 

 1. 如果在编写代码时, 指导类的名称, 且类已经存在, 可以通过 包名.类名.class 得到一个类的 类对象

2. 如果拥有类的对象, 可以通过 Class 对象.getClass() 得到一个类的 类对象

3. 如果在编写代码时, 知道类的名称 , 可以通过 Class.forName(包名+类名): 得到一个类的 类对象(后期写动态用的最多)可以不先创建类,比如需要先得到class,而需要执行的代码过两天才需要写入就可以选择这种方式。

上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存 ! 如果类已经在内存中存在, 不 会重复加载, 而是重复利用 !

package kaikeba.Demo;

public class fanshe {
    public static void main(String[] args) throws ClassNotFoundException {
//        第一种将类加载到内存的方式
        Class<Person> p = Person.class;
        System.out.println(p);
        //        第二种将类加载到内存的方式
        Person p1 = new Person();
        Class<Person> c = (Class<Person>)p1.getClass();
//        第三种将类加载到内存的方式:可以不先创建Person类。
        Class<Person> c2 =  (Class<Person>)Class.forName("kaikeba.Demo.Person");
        System.out.println(c2);
        System.out.println(p == c&& p==c2);


    }
//    未来写框架也会用到这种方式,只需传名称就可获得。
//    public static Class getClass(String className) throws ClassNotFoundException {
//        return Class.forName(className);
//    }
}

3.通过class对象 获取一个类的构造方法

3.1Constructor 创建对象

  常用方法:

  newInstance(Object... para) 调用这个构造方法, 把对应的对象创建出来

  参数: 是一个Object类型可变参数, 传递的参数顺序 必须匹配构造方法中形式参数列表的顺 序!

  setAccessible(boolean flag) 如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

3.2获取一个类的构造方法

1. 通过指定的参数类型, 获取指定的单个构造方法 getConstructor(参数类型的class对象数   组) 例如: 构造方法如下: Person(String name,int age) 得到这个构造方法的代码如下:

   Constructor c = p.getClass().getConstructor(String.class,int.class);

2. 获取构造方法数组 getConstructors();

3. 获取所有权限的单个构造方法 getDeclaredConstructor(参数类型的class对象数组)

4. 获取所有权限的构造方法数组 getDeclaredConstructors();

案例

package fanshe.gouzao;

public class Person {
    private String name;
    private  int age;

    private  Person(String name){
        this.name = name;
        this.age = 88;
    }

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

package fanshe.gouzao;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Demo_1 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
        Class<Person> pClass = (Class<Person>) Class.forName("fanshe.gouzao.Person");
//        找到无参的构造方法
        Constructor <Person> c1 = pClass.getConstructor();
//        使用无参的构造方法,创建对象
        Person p = c1.newInstance();
        System.out.println(p);

//        找到一个含有String name 和int age 的构造方法
      Constructor<Person> c2 =  pClass.getConstructor(String.class,int.class);
//      使用这个构造方法,创建对象
//        用的是两参的构造方法,如果不传参数执行时会报错
        Person p1 =c2.newInstance("张三",12);
        System.out.println(p1);


    Constructor <Person> c3 = pClass.getDeclaredConstructor(String.class);
    //    忽略权限检查
        c3.setAccessible(true);
    Person p3 = c3.newInstance("李四");

        System.out.println(p3);
    }
}

4、获取Method

4.1、通过class对象 获取一个类的方法

1. getMethod(String methodName , class.. clss) 根据参数列表的类型和方法名, 得到一个方法(public修饰的)

2. getMethods(); 得到一个类的所有方法 (public修饰的)

3. getDeclaredMethod(String methodName , class.. clss) 根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)

4. getDeclaredMethods(); 得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)

4.2、Method 执行方法

invoke(Object o,Object... para) : 调用方法 ,

    参数1. 要调用方法的对象

    参数2. 要传递的参数列表

getName() 获取方法的方法名称

setAccessible(boolean flag)

    如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

案例

package fangfa.fanshe;

public class Person {
    private String name;
    private  int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

   private void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

package fangfa.fanshe;

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

public class Demo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
//        加载类
        Class c1 = Class.forName("fangfa.fanshe.Person");
//        获取构造方法
        Constructor c = c1.getConstructor();
//        创建对象
        Object o = c.newInstance();
//        获取类的方法
        Method setName = c1.getMethod("setName",String.class);
//        setAge方法已私有,getMethod()无法找到该权限的方法,但getDeclaredMethod()可以找到任何权限下的方法
        Method setAge = c1.getDeclaredMethod("setAge",int.class);
//        忽视权限检查
        setAge.setAccessible(true);
//        参数一.那个对象要执行setName方法
//        参数二.调用方法时传递的参数0-n
        setName.invoke(o,"张三");
        setAge.invoke(o,18);
        System.out.println(o);
    }
}

5、获取Field

5.1、通过class对象 获取一个类的属性

1. getDeclaredField(String filedName) 根据属性的名称, 获取一个属性对象 (所有属性)

2. getDeclaredFields() 获取所有属性

3. getField(String filedName) 根据属性的名称, 获取一个属性对象 (public属性)

4. getFields() 获取所有属性 (public)

5.2、Field 属性的对象类型 

常用方法:

  1. get(Object o ); 参数: 要获取属性的对象 获取指定对象的此属性值

  2. set(Object o , Object value);

        参数1. 要设置属性值的 对象

        参数2. 要设置的值 设置指定对象的属性的值

  3. getName() 获取属性的名称

  4. setAccessible(boolean flag) 如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的属性)

案例

package shuxing;

public class Person {
    private  String name;
    private  int age;
    public String  number;

    public Person() {
    }

    public Person(String name, int age, String number) {
        this.name = name;
        this.age = age;
        this.number = number;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getNumber() {
        return number;
    }

    public void setNumber(String number) {
        this.number = number;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", number='" + number + '\'' +
                '}';
    }
}
package shuxing;

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

public class Demo1 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException, NoSuchFieldException {
//        加载类
       Class c = Class.forName("shuxing.Person");
//       获得构造方法
      Constructor ct = c.getConstructor();
//      创建对象
        Object o = ct.newInstance();

//        获取不带权限的属性属性
        Field phone = c.getField("number");
       phone.set(o,"15515566666");
//获取代有权限的属性
       Field name = c.getDeclaredField("name");
       name.setAccessible(true);
       name.set(o,"网二");


        System.out.println(o);



    }
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值