栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

深入理解Java反射机制

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

深入理解Java反射机制

一、Java反射定义

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


指在Java程序运行状态中,

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

这种动态获取类的内容以及动态调用对象的方法和获取属性的机制.就叫做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() 得到接口
ClassgetSuperclass() 得到父类
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";
    }

    
    public static Office getInstanceByKey(String key){
        if("word".equals(key)){
            return new Word();
        }

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

    
    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);
}

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 aClass = s1.getClass();
    Constructor 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("实例已经创建,不允许再创建了...");
    }
}

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/694728.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号