IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 深入理解Java反射机制 -> 正文阅读

[Java知识库]深入理解Java反射机制

一、Java反射定义

反射即反向探知,有点像考古学家根据发掘的物品来探知以前的事情

在这里插入图片描述
指在Java程序运行状态中,

  1. 对于给定的一个类(Class)对象,可以获得这个类(Class)对象的所有属性和方法;
  2. 对于给定的一个对象(new XXXClassName<? extends Object>),都能够调用它的任意一个属性和方法.

这种动态获取类的内容以及动态调用对象的方法和获取属性的机制.就叫做JAVA的反射机制
如下案例

package com.example.demo;

public class Person {
    public void say(){
        System.out.println("Person中的say方法");
    };
}
public static void main(String[] args) throws Exception {
        // 获取类对象
        Class clazz = Person.class;
        // 创建对象
        Person person = (Person) clazz.newInstance();
        // 获取类的相关结构
        System.out.println(clazz.getName()); // 类名
        System.out.println(clazz.getPackage()); // 包名
        System.out.println(clazz.getClassLoader()); // 获取类的classLoader
        System.out.println(clazz.getSuperclass()); // 获取父类

        // 获取方法
        Method method = clazz.getDeclaredMethod("say");
        // 等价于 Person p = new Person();
        // p.say();
        method.invoke(person);

    }

在这里插入图片描述

通过反射对象得到构造方法\成员变量\成员方法

Constructor<?>[]getConstructors() 得到构造方法
Field[]getDeclaredFields() 得到成员变量
Method[]getDeclaredMethods() 得到成员方法
Class<?>[]getInterfaces() 得到接口
Class<? super T>getSuperclass() 得到父类
PackagegetPackage() 得到包对象
intgetModifiers() Java语言修饰符 Modifier的toString(int mod)
StringgetName() 得到类名称

二、Java反射的优缺点

优点
增加程序的灵活性,避免将固有的逻辑程序写死到代码里
代码简洁,可读性强,可提高代码的复用率
缺点
相较直接调用在量大的情景下反射性能下降
内部暴露和安全隐患

public interface Office {

    void toPDF();
}

public class Word implements Office {
    @Override
    public void toPDF() {
        System.out.println("Word 2 PDF ");
    }
}

public class Excel implements Office {
    @Override
    public void toPDF() {
        System.out.println(" Excel 2 PDF");
    }
}

package com.example.demo.fashe1;

public class Main {
    public static void main(String[] args) {
        String key = "word";
    }

    /**
     * 通过传入的Key,返回不同的对象
     * @param key
     * @return
     */
    public static Office getInstanceByKey(String key){
        if("word".equals(key)){
            return new Word();
        }

        if("excel".equals(key)){
            return new Excel();
        }
        return null;
    }

    /**
     * 通过反射机制来动态创建
     * @param key
     * @return
     */
    public static Office getInstanceReflectByKey(String key){
        String packageName = "com.example.demo.fashe1";
        Office office = null;
        try {
            Class clazz = Class.forName(packageName+"."+key);
            office = (Office)clazz.newInstance();
        }catch (Exception e){
            e.printStackTrace();
        }
        return office;
    }
}

三、反射到底慢在哪些地方

寻找类Class字节码的过程
安全管理机制的权限验证等等
若需要调用native方法调用时JNI接口的使用

public static void main(String[] args) {
    String key = "word";
    //getInstanceByKey("word");
    long startTime = System.currentTimeMillis();
    for (int i = 0; i < 1000000 ; i++) {
        getInstanceByKey(key);
    }
    long endTime = System.currentTimeMillis();
    System.out.println("总计花费时间:" + (endTime - startTime));
}

通过new的方式创建1000000个对象只需要花费9毫秒

在这里插入图片描述
如果通过反射的方式的话

public static void main(String[] args) {
    String key = "Word";
    //getInstanceByKey("word");
    long startTime = System.currentTimeMillis();
    for (int i = 0; i < 1000000 ; i++) {
        //getInstanceByKey(key);
        getInstanceReflectByKey(key);
    }
    long endTime = System.currentTimeMillis();
    System.out.println("总计花费时间:" + (endTime - startTime));
}

在这里插入图片描述
大家能发现差距还是非常大的~

源码层面

@CallerSensitive
public static Class<?> forName(String className)
            throws ClassNotFoundException {
    Class<?> caller = Reflection.getCallerClass();
    return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}

/** Called after security check for system loader access checks have been made. */
private static native Class<?> forName0(String name, boolean initialize,
                                        ClassLoader loader,
                                        Class<?> caller)
    throws ClassNotFoundException;

在这里插入图片描述
在这里插入图片描述

四、Class内部

在这里插入图片描述
在编写每个类实例中,都会定义这个类的包名,类名,访问域,特征符,构造器,字段,函数,父类,接口等等内容.

这些内容在我们的Class类中都提供了对应的获取方法进行获取.

//clazz 代表对应的Class类实例

五、反射的基本操作

5.1 获取类对象的四种方式
Class clazz = Person.class;
Class clazz2 = new Person().getClass();
Class clazz3 = Class.forName("com.example.demo.fashe.Person");
Class clazz4 = Demo02.class.getClassLoader().loadClass("com.example.demo.fashe.Person");

5.1.2 获取类对象的四种方式
// 获取类的相关结构
int modifier = clazz.getModifiers(); // 获取类修饰符
Package aPackage = clazz.getPackage(); // 获取类包名
String fullClassName = clazz.getName(); // 获取类的全路径名称
String simpleName = clazz.getSimpleName(); // 获取类的简单名称
ClassLoader classLoader = clazz.getClassLoader(); // 获取类加载器
Class[] interfaces = clazz.getInterfaces(); // 获取类实现的接口列表
Class superclass = clazz.getSuperclass(); // 获取类的父类
Annotation[] annotations = clazz.getAnnotations(); // 获取类的注解信息

在这里插入图片描述

5.2 类的属性操作
Person person = (Person) clazz.newInstance();
// 获取类中所有的共有字段 包含继承的字段
Field[] fields = clazz.getFields();
// 获取类中定义的字段 内部
Field[] declaredFields = clazz.getDeclaredFields();
// 获取指定名称的类中定义的字段
Field nameField = clazz.getDeclaredField("name");
// 获取字段的修饰符
int modifiers = nameField.getModifiers();
// 指定字段强制访问
nameField.setAccessible(true);
// 修改字段的值
nameField.set(person,"平安");
// 静态字段赋值
nameField.set(null,"静态字段赋值");

5.3 类的方法操作
// 获取类中的所有的共有的方法 继承
Method[] methods = clazz.getMethods();
// 获取类中的定义的方法
Method[] declaredMethods = clazz.getDeclaredMethods();
// 获取类中指定名称和参数的公有方法
Method say = clazz.getMethod("say", String.class);
// 获取类中定义的指定名称和参数的方法
Method say1 = clazz.getDeclaredMethod("say");
// 获取方法的修饰符
int modifiers1 = say.getModifiers();
// 指定对象进行成员方法的调用
Object 小米666 = say.invoke(person, "小米666");
say.setAccessible(true);// 指定方法的强制执行
// 静态方法调用
say.invoke(null);

5.4 构造器的操作
Constructor[] cons = clazz.getConstructors();            //获取类中所有的公有构造器
Constructor[] cons1 = clazz.getDeclaredConstructors();       //获取类中所有的构造器
Constructor conNoParam= clazz.getDeclaredConstructor();       //获取类中无参的构造器
Constructor con= clazz.getDeclaredConstructor(String.class,String.class);   //获取类中有参构造
int modifers = con.getModifiers();             //获取构造器的修饰符
conNoParam.newInstance();              //构造器实例对象
con.setAccessible(true);                  //指定方法的强制访问
con.newInstance("abc","bbb");              //有参构造调用
Person.class.newInstance();                //class直接调用默认无参构造

newInstance();方法的本质

在这里插入图片描述

六、反射破坏了单例模式

定义一个简单的单例模式

public class PersonSingle {

    private static PersonSingle instance;

    private PersonSingle(){

    }

    public static synchronized  PersonSingle getInstance(){
        if(instance == null){
            instance = new PersonSingle();
        }
        return instance;
    }
}

通过反射可以创建多个实例,从而破坏单例的设计

public static void main(String[] args) throws Exception{
    PersonSingle s1 = PersonSingle.getInstance();
    System.out.println(s1);
    Class<? extends PersonSingle> aClass = s1.getClass();
    Constructor<? extends PersonSingle> declaredConstructor = aClass.getDeclaredConstructor();
    declaredConstructor.setAccessible(true);
    Thread.sleep(1000);
    // 通过反射 调用私有的构造器来创建对象,从而破坏单例设计
    PersonSingle s2 =  declaredConstructor.newInstance(null);
    System.out.println(s2);
}

在这里插入图片描述
那么如何防止这种破坏呢,其实很简单我们只需要在私有构造中加个判断就可以了,如下

private  PersonSingle(){
    if(PersonSingle.instance != null){
        throw new RuntimeException("实例已经创建,不允许再创建了...");
    }
}

在这里插入图片描述

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-01-04 13:16:53  更:2022-01-04 13:19:34 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 7:16:51-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码