java-反射机制

简述

反射机制允许程序在执行期间借助于Reflection API取得任何类的内部信息(如成员变量,构造器,成员方法等),并能操作对象的属性及方法。反射机制在设计模式和框架底层都能用到。

        类一旦加载,在堆中会产生一个Class类型的对象,这个对象包含了类的完整结构信息。通过这个对象可以获得这个类的结构

java反射机制相关类

         java作为面向对象的编程语言,万物皆对象,在反射机制中,将类的各个成员都当做一个类的对象

    ①java.lang.Class:Class对象表示某个类加载后在堆中的对象
    ②java.lang.reflect.Method:代表类的方法
    ③java.lang.reflect.Field:代表类的成员变量
    ④java.lang.reflect.Constructor:代表类的构造器

反射机制优缺点 

优点:可以动态的创建和使用对象,使用灵活,反射机制就是框架的底层支撑
缺点:使用反射的运行效率较慢

 反射机制优化

    关闭访问检查,Method,Field,Constructor均有setAccessible(boolean b)方法,作用是开启或关闭访问安全检查开关,参数为true时关闭访问检查,提高反射的效率;参数为false则表示开启访问检查

Class 

①Class类也是一个类,只是类名就叫做Class,因此也继承Object类
②Class类对象不是new Class()的形式创建的,而是由系统创建
③对于某一个类的Class类对象,在内存中只有一份,因此类在内存中只加载一次
④通过Class类对象可以获取所加载类的所有接信息
⑤Class对象存放在堆中。

具有Class对象的类型 

类,接口,数组,枚举,注解,基本数据类型,void

代码示例:


package com.flash.class_;
 
import java.io.Serializable;
 
/**
 * @author flash
 * @date 2024/06/18 18:08
 * 功能描述:枚举拥有 Class 对象的类型
 */
public class AllTypeClass {
    public static void main(String[] args) {
        // 类
        Class<String> cls1 = String.class;
        // 接口
        Class<Serializable> cls2 = Serializable.class;
        // 一维数组
        Class<Integer[]> cls3 = Integer[].class;
        // 二维数组
        Class<float[][]> cls4 = float[][].class;
        // 注解
        Class<Deprecated> cls5 = Deprecated.class;
        // 枚举
        Class<Thread.State> cls6 = Thread.State.class;
        // 基本数据类型
        Class<Long> cls7 = long.class;
        // void
        Class<Void> cls8 = void.class;
        // Class
        Class<Class> cls9 = Class.class;
 
        System.out.println("cls1 = " + cls1);
        System.out.println("cls2 = " + cls2);
        System.out.println("cls3 = " + cls3);
        System.out.println("cls4 = " + cls4);
        System.out.println("cls5 = " + cls5);
        System.out.println("cls6 = " + cls6);
        System.out.println("cls7 = " + cls7);
        System.out.println("cls8 = " + cls8);
        System.out.println("cls9 = " + cls9);
    }
}

 类加载

基本说明:反射机制是java实现动态语言的关键, 也就是通过反射实现类动态加载
        ①静态加载:编译时加载相关的类, 如果没有直接报错, 依赖性强
        ②动态加载:运行时加载需要的类, 如果运行时没有用到该类, 即使不存在这个类也不会报错, 依赖性较弱
类加载时机:
        ①当创建对象时(new) 静态加载
        ②当子类被加载时, 父类也被加载 静态加载
        ③调用类中的静态成员时 静态加载
        ④反射机制 动态加载

Class类对象创建方式 

package com.ffyc.javareflect;

public class Test1 {
    public static void main(String[] args) throws ClassNotFoundException {
        //以前使用类,已知类名:new创建对象,调用对象成员
        // User user =new User();
        //user.eat();

        //1使用反射机制时,只知道类的名称(包名+类名)
        String classname  = "com.ffyc.javareflect.User";
        //如何类的信息?可以通过class 类来获得类中的信息

        //如何获得类的Class对象?
        // 方式1:

        Class clazz1 = Class.forName(classname);
        System.out.println(clazz1);

        //方式2:
        Class clazz2 = User.class;
        System.out.println(clazz1==clazz2);

        //方式3:
        User user = new User();
        Class clazz3 = user.getClass();
        System.out.println(clazz1==clazz3);
    }
}

运行结果: 

反射获取类的结构信息

代码示例:

class PersonDad {
    public String dept;
 
    public PersonDad() {
    }
 
    public PersonDad(String dept) {
        this.dept = dept;
    }
 
    public void dadMethod() {
    }
}
 
interface PersonInterface {
 
}
 
@Deprecated // 弃用注解
class Person extends PersonDad implements PersonInterface {
    // 属性
    public String name = "JSON";
    protected int age = 20;
    String job = "study";
    private double sal = -100;
    static int a;
    final int b = 10;
 
    public Person() {
    }
 
    public Person(String name) {
        this.name = name;
    }
 
    private Person(String dept, String job) {
        super(dept);
        this.job = job;
    }
 
 
    protected Person(String name, String job, String dept) {
        super(dept);
        this.name = name;
        this.job = job;
    }
 
    public Person(String dept, String name, int age, String job, double sal) {
        super(dept);
        this.name = name;
        this.age = age;
        this.job = job;
        this.sal = sal;
    }
 
    // 方法
    public void m1(String name, int age) {
    }
 
    protected int m2() {
        return 0;
    }
 
    String m3() {
        return null;
    }
 
    private void m4() {
    }
 
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", job='" + job + '\'' +
                ", sal=" + sal +
                ", b=" + b +
                '}';
    }
}

获取方法

package com.ffyc.javareflect;

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

public class Test2 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
                /*
        java中创建对象的方式:
        1.new
        2.反序列化
        3.反射机制

        */
        //使用反射机制时,只知道类的名称(包名+类名)
        String classname ="com.ffyc.javareflect.User";
        //string classname ="com.ffyc.javareflect.Car";

        //1.通过类名 获得到类的class 对象
        Class aClass =Class.forName(classname);

        //2.通过类的class 对象  创建对象
        Object object = aClass.newInstance();
        System.out.println(object);

        //获得类中的构造方法   通过构造方法api中的方法创建对象
        Constructor constructor1 = aClass.getConstructor();;//获得指定的公共构造方法
        Object object1 = constructor1.newInstance();
        Constructor constructor2 = aClass.getConstructor(String.class,String.class);

        Object object2 = constructor2.newInstance("张三","111");
        System.out.println(object1);
        System.out.println(object2);

        Constructor [] constructors = aClass.getConstructors();//获得所有的公共构造方法
        System.out.println(constructors.length);

        //虽然可以获得私有构造方法 但一般不建议操作 因为打破了封装性
        aClass.getDeclaredConstructor();//获得类中任意的构造方法  包含私有的
        System.out.println(aClass.getDeclaredAnnotations().length);

    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值