Java|Java 反射(Reflect)详解

目录

  • 一 首先我们的去知道什么是反射?
  • 二(刨根问底)知道是什么还需要知道什么“成分”组成反射?
    • 2.1 Class 对象的获取及使用
    • 2.2 拿到碗筷就得去盛饭,拿到了Class就得去操作,获得属性
    • 2.3 吃饱饭,我还想学做饭,找到Class,但是我想去获取Class对象的实例。
    • 2.4 调用Class的实例对象的方法
    • 2.5 修改类的私有属性,由于是私有属性,所以需要去关闭程序的安全监测。
  • 三反射的性能
    • 3 .1 走了不同的路,就得比较不同路的对比(性能对比);
  • 四反射的的优缺点
    • 4.1反射的优点
    • 4.2反射的缺点
  • 五 反射的使用场合
    • 总结

      一 首先我们的去知道什么是反射? 加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象 ,一个类只有一个Class对象,这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,通过这个镜子可以看到类的结构,所以我们形象的称为:反射 ,或者说动态(运行时)获取类信息和调用类方法

      二(刨根问底)知道是什么还需要知道什么“成分”组成反射? 想要去实现反射想要借助一些类分别是 class,Constructor,Field,Method;

      Java|Java 反射(Reflect)详解
      文章图片


      Java|Java 反射(Reflect)详解
      文章图片


      2.1 Class 对象的获取及使用

      吃饭的先拿碗筷,反射就得先找先找 Class

      找Class的有三种方式
      1实例化对象获取该实例的 Class 对象

      Person person = new Student();

      Class s=person.getClass();
      2通过类的全限定名获取该类的 Class 对象

      Class s1 =Class.forName(“com.ma.reflect.Student”);
      3通过类名.class

      Class S2=Student.class;

      2.2 拿到碗筷就得去盛饭,拿到了Class就得去操作,获得属性
      代码案例
      package com.ma.reflect; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; public class reflectTest6{public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {//反射Class c=Class.forName("com.ma.reflect.User"); System.out.println("获得包+类"+c.getName()); //获得包+类System.out.println("获得类名"+c.getSimpleName()); //获得类名//获得类的属性System.out.println("-------------------------"); Field[]fields=c.getFields(); //的到public类for (Field field : fields) {System.out.println("获得类的Public 属性"+field); }System.out.println("-------------------------"); //获得全部的类属性Field[] f1=c.getDeclaredFields(); for (Field field : f1) {System.out.println("获得类的全部属性"+field); }System.out.println("-------------------------"); //获得指定属性的值Field name=c.getDeclaredField("age"); System.out.println("获得指定了属性的类型"+name); System.out.println("-------------------------"); //获得指定方法的值Method[] methods=c.getMethods(); System.out.println("获得所有的public方法"); for (Method method : methods) {System.out.println("public类有"+method); }Method[] methods1=c.getDeclaredMethods(); System.out.println("获得所有的public方法"); for (Method method : methods1) {System.out.println("所有的方法="+method); }System.out.println("-------------------------"); System.out.println("获得所有的public构造器"); Constructor[] constructors=c.getConstructors(); for (Constructor constructor : constructors) {System.out.println("public的构造器"+constructor); }System.out.println("获得所有的构造器"); Constructor[] constructors1=c.getDeclaredConstructors(); for (Constructor constructor : constructors1) {System.out.println("所有的构造器"+constructor); }}}

      Java|Java 反射(Reflect)详解
      文章图片


      Java|Java 反射(Reflect)详解
      文章图片


      【Java|Java 反射(Reflect)详解】Java|Java 反射(Reflect)详解
      文章图片


      Java|Java 反射(Reflect)详解
      文章图片


      2.3 吃饱饭,我还想学做饭,找到Class,但是我想去获取Class对象的实例。
      Class user=Class.forName("com.ma.reflect.User"); //获得Class对象//构造对象User user1=(User) user.newInstance(); //无参方法System.out.println(user1); //通过构造器 创建有参方法Constructor constructor=user.getDeclaredConstructor(String.class,int.class,int.class); User user2= (User) constructor.newInstance("myt",28,213); System.out.println(user2);

      Java|Java 反射(Reflect)详解
      文章图片



      2.4 调用Class的实例对象的方法
      Class user=Class.forName("com.ma.reflect.User"); //获得Class对象User user3=(User) user.newInstance(); //通反射去获取一个方法 Method setName=user.getDeclaredMethod("setName", String.class); //invoke激活 (对象,参数)setName.invoke(user3,"myt"); System.out.println(user3.getName());

      Java|Java 反射(Reflect)详解
      文章图片



      2.5 修改类的私有属性,由于是私有属性,所以需要去关闭程序的安全监测。

      //通过反射去调用方法
      User user4=(User) user.newInstance(); Field name=user.getDeclaredField("name"); //得到属性//***修改权限 去实现对私有属性的修改 通过关闭程序的安全监测name.setAccessible(true); name.set(user4,"tym"); //private的属性无法直接去调用并且修改System.out.println(user4.getName());

      Java|Java 反射(Reflect)详解
      文章图片


      三反射的性能
      3 .1 走了不同的路,就得比较不同路的对比(性能对比);

      普通方法、反射、关闭安全检测的反射
      public class reflectTest8 {//普通方法调用publicstaticvoidtest(){User user=new User(); Long startTime=System.currentTimeMillis(); for (int i = 0; i < 100000000; i++) {user.getName(); }Long endTime=System.currentTimeMillis(); System.out.println("普通方法"+(endTime-startTime)+"ms"); }//反射方法调用public staticvoidtest1() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {User user=new User(); Class c=Class.forName("com.ma.reflect.User"); Method getName=c.getDeclaredMethod("getName",null); Long startTime=System.currentTimeMillis(); for (int i = 0; i < 100000000; i++) {getName.invoke(user,null); }Long endTime=System.currentTimeMillis(); System.out.println("反射方法"+(endTime-startTime)+"ms"); }//反射方法 关闭检测调用public staticvoidtest2() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {User user=new User(); Class c=Class.forName("com.ma.reflect.User"); Method getName=c.getDeclaredMethod("getName",null); getName.setAccessible(true); Long startTime=System.currentTimeMillis(); for (int i = 0; i < 100000000; i++) {getName.invoke(user,null); }Long endTime=System.currentTimeMillis(); System.out.println("反射方法 关闭检测"+(endTime-startTime)+"ms"); }public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException {test(); test1(); test2(); }}

      Java|Java 反射(Reflect)详解
      文章图片


      四反射的的优缺点
      4.1反射的优点
      反射提高了Java程序的灵活性和扩展性,降低耦合性,提高自适应能力。它允许程序创建和控制任何类的对象,无需提前硬编码目标类

      4.2反射的缺点
      因为是JVM操作,所以对于性能来说会有所下降。
      容易对程序源码造成一定的混乱。

      五 反射的使用场合 在编译时根本无法知道该对象或类可能属于哪些类,程序只依靠运行时信息来发现该对象和类的真实信息

      总结 本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

        推荐阅读