详解JAVA 反射机制

 更新时间:2020年06月16日 15:15:59   作者:追风的小蚂蚁  
这篇文章主要介绍了JAVA 反射机制的相关知识,文中讲解的非常细致,代码帮助大家更好的理解学习,感兴趣的朋友可以了解下

什么是反射?

反射机制是在程序运行状态中,对于任意一个类,都能够获取这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取信息以及动态调用对象的方法的功能称为java语言的反射机制。

反射的作用

1.可以实现简单的反编译,获取类中的属性和方法等基本信息,.class—>java

2.通过反射机制获取类的属性、方法等

在使用eclipse时,通过对象引用.的方式,eclipse就会将这个对象中的所有属性和方法展示出来,这个就是利用的反射机制。其实反射应用最多的地方就是将来要学习的框架,例如spring、spring MVC、Mybatis、hibernate、Struts等等

反射的优点

提高程序的灵活性和扩展性。使用反射机制后,程序在创建对象的时候不用使用new关键字写死在程序中了,而是从配置文件中进行读取,这样可以提高程序的灵活性

反射的缺点

影响性能。使用反射时,代码量稍微多点,并且是要动态的将读取类,所以会影响性能。下面例子可能不太恰当,不过希望能够帮助你更好地了解,比如你要做汽车:

正常的方式是司机停车,你开门上车。
动态的方式就是司机不停车,只降低车速,然后你跑着开门上车。

破坏封装性。

Class对象

在类加载器将.class文件读取到内存中的时候,jvm会创建这个.class文件的对象,并且只创建一个存放到jvm的方法区内存中,在java.lang包下有个Class类,这个类就是.class文件的对象类型,任何类在被使用时,都会创建这个类的Class对象。除此之外,在java.lang.reflect包下面的类也跟反射有关。
创建一个Person类:

package com.sutaoyu.reflect;

public class Person {
  private String name;
  public Person() {
    System.out.println("Person类的构造方法");
  }
  
  public Person(String name) {
    this.name = name;
  }
  
  public void sing() {
    System.out.println("唱歌");
  }
  
  public void setName(String name) {
    this.name = name;
  }
  
  public String getName() {
    return name;
  }
}

获取这个Person类的Class对象有三种方式:

//第一种方式:
//c1引用的对象代表整个Person类
Class c1 = Class.forName("com.monkey1024.reflect.Person");

//第二种方式:
//java中每个类型都有 class 属性.
Class c2 = Person.class;

//第三种方式:
//java语言中任何一个java对象都有getClass 方法
Person p = new Person();
Class c3 = e.getClass(); 

//因为Person这个类在JVM中只有一个,所以c1,c2,c3的内存地址是相同的,指向堆中唯一的Class对象.
System.out.println(c1==c2); //true
System.out.println(c2==c3); //true

使用反射将.class文件读取到内存中

将上面的Person.java文件删除,留下Person.class文件。

使用反射将Person.class文件读取到内存中

package com.sutaoyu.reflect;

public class RefectTest02 {
  public static void main(String[] args) {
    try {
      //读取时需要加上类的包名
      Class clazz = Class.forName("com.sutaoyu.reflect.Person");
      Object o = clazz.newInstance();
      System.out.println(o);
    }catch(ClassNotFoundException e) {
      e.printStackTrace();
    }catch(InstantiationException e) {
      e.printStackTrace();
    }catch(IllegalAccessException e) {
      e.printStackTrace();
    }
  }
}

使用反射获取类中的属性

下面程序在获取类中的属性之后,将.class文件中的属性反编译并打印出来了。

package com.sutaoyu.reflect;

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;

public class ReflectTest04 {
  public static void main(String[] args) throws ClassNotFoundException {
    Class c = Class.forName("java.lang.Integer");
    //获取类中所有的属性
    Field[] field = c.getDeclaredFields();
    
     //使用反射反编译
    StringBuilder sb = new StringBuilder(200);
    sb.append(Modifier.toString(c.getModifiers()) + " class "+c.getSimpleName() + "{\n");
    //访问权限修饰符
    String s = Modifier.toString(f.getModifiers());
    if(!"".equals(s)) {
      sb.append(Modifier.toString(f.getModifiers()) + ""); 
    }
    
    //属性的类型名称
    Class type = f.getType();
    sb.append(f.getName() + ";\n");
    
    //属性的名字
    sb.append("}");
    
    System.out.println(sb.toString());
    
  }
}

使用反射获取类中指定的属性并赋值

package com.sutaoyu.reflect;

import java.lang.reflect.Field;

/**
 * 使用反射获取类中指定的属性,并且为其赋值
 *
 */
public class ReflectTest05 {

  public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException {
    Class c = Class.forName("com.monkey1024.reflect.User");
    /*//获取指定属性的Field对象
    Field f = c.getDeclaredField("name");
    //创建对象
    Object o = c.newInstance();

    //给o对象上的name属性赋值为张三
    f.set(o, "张三");
    //获取o对象上的name属性的值
    System.out.println(f.get(o));*/

    Field f = c.getDeclaredField("age");
    Object o = c.newInstance();
    //从外部打破封装性
    f.setAccessible(true);
    f.set(o, 20);
    System.out.println(f.get(o));
  }

}

使用反射获取类中的方法

下面程序将.class文件中的方法反编译并打印出来了

User类:

package com.sutaoyu.reflect;

import java.util.Date;

public class User {

  private int age;
  public String name;
  protected Date birthday;
  boolean sex;

  public void m1(){

  }

  public static void m2(){

  }

  private String m3(){
    return "";
  }

  public void m4(int i, String s){

  }

  public String m5(String s, int i){
    return s + " , " + i;
  }
}

测试类:

package com.sutaoyu.reflect;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

/**
 * 使用反射获取类中的方法
 *
 */
public class ReflectTest06 {

  public static void main(String[] args) throws ClassNotFoundException {
    //Class c = Class.forName("com.monkey1024.reflect.User");
    Class c = Class.forName("java.lang.Object");
    //获取类中所有方法
    Method[] method = c.getDeclaredMethods();
    /*for(Method m : method){
      //方法修饰符
      System.out.println(Modifier.toString(m.getModifiers()));
      //方法的返回值类型
      Class type = m.getReturnType();
      System.out.println(type.getSimpleName());
      //方法名
      System.out.println(m.getName());
      //方法参数
      Class[] param = m.getParameterTypes();
      for(Class p : param){
        System.out.println(p.getSimpleName());
      }
    }*/

    //反编译将User类中的方法打印
    StringBuilder sb = new StringBuilder(200);
    sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() + "{\n");
    for(Method m : method){
      sb.append("\t");
      //方法的修饰符
      sb.append(Modifier.toString(m.getModifiers()) + " ");
      //方法的返回值类型
      Class type = m.getReturnType();
      sb.append(type.getSimpleName() + " ");
      //方法名
      sb.append(m.getName() + " ");

      //方法参数
      sb.append("(");
      Class[] param = m.getParameterTypes();
      for(int i=0; i<param.length; i++){
        if(i == param.length-1){
          sb.append(param[i].getSimpleName());
        }else{
          sb.append(param[i].getSimpleName());
          sb.append(" ,");
        }
      }
      sb.append(")");
      sb.append("{}\n");

    }
    sb.append("}");

    System.out.println(sb.toString());
  }

}

使用反射调用类中的方法

package com.sutaoyu.reflect;

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

/**
 * 使用反射调用类中的方法
 *
 */
public class ReflectTest07 {

  public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    Class c = Class.forName("com.monkey1024.reflect.User");
    //获取方法
    Method method = c.getDeclaredMethod("m5", String.class, int.class);
    //创建对象
    Object o = c.newInstance();
    Object result = method.invoke(o, "admin", 10);
    System.out.println(result);
  }

}

使用反射获取构造方法

package com.sutaoyu.reflect;

import java.lang.reflect.Constructor;
import java.lang.reflect.Modifier;

/**
 * 使用反射获取类中的构造方法
 *
 */
public class ReflectTest08 {

  public static void main(String[] args) throws ClassNotFoundException {
    Class c = Class.forName("java.lang.StringBuffer");
    //获取类中所有的构造方法
    Constructor[] con = c.getDeclaredConstructors();
    for(Constructor co : con){
      //获取修饰符
      System.out.println(Modifier.toString(co.getModifiers()));

      //获取方法名
      System.out.println(co.getName());

      //获取方法参数
      Class[] type = co.getParameterTypes();
      for(Class t : type){
        System.out.println(t.getSimpleName());
      }
    }
  }

}

使用反射获取父类和父接口

package com.monkey1024.reflect;

/**
 * 使用反射获取父类和父接口
 *
 */
public class ReflectTest09 {

  public static void main(String[] args) throws ClassNotFoundException {
    Class c = Class.forName("java.lang.StringBuffer");
    //获取父类
    Class sup = c.getSuperclass();
    System.out.println(sup.getName());

    //获取父接口
    Class[] inter = c.getInterfaces();
    for(Class i : inter){
      System.out.println(i.getName());
    }
  }

}

以上就是详解JAVA 反射机制的详细内容,更多关于JAVA 反射机制的资料请关注脚本之家其它相关文章!

相关文章

  • 解决IDEA使用maven创建Web项目,出现500错误的问题

    解决IDEA使用maven创建Web项目,出现500错误的问题

    本文主要介绍了在使用Maven创建项目并导入依赖写完测试代码后运行出现500错误的解决步骤,这种问题的根本原因是Tomcat启动后缺少某些支持的jar包,导致运行出错,解决方法是在项目结构中找到Artifacts,点击要编辑的项目
    2024-10-10
  • Spring避免循环依赖的策略详解

    Spring避免循环依赖的策略详解

    在Spring框架中,循环依赖是指两个或多个bean相互依赖对方,形成一个闭环,这在应用启动时可能导致BeanCurrentlyInCreationException异常,本文给大家介绍了Spring中如何避免循环依赖,需要的朋友可以参考下
    2024-02-02
  • 如何获取 Spring heapdump中的明文密码

    如何获取 Spring heapdump中的明文密码

    Actuator是Spring Boot提供的应用系统监控的开源框架,在攻防场景里经常会遇到Actuator配置不当的情况,攻击者可以直接下载heapdump堆转储文件,本文介绍如何获取 Spring heapdump中的密码明文,需要的朋友可以参考下
    2022-07-07
  • Java中valueOf和parseInt的区别详解

    Java中valueOf和parseInt的区别详解

    这篇文章主要介绍了Java中valueOf和parseInt的区别详解,在编程中,遇到类型转换,好像会经常用到 parseInt 和 valueOf,当然这里只拿 Integer 类型进行陈述,其他类型也是雷同的,需要的朋友可以参考下
    2024-01-01
  • JAVA中HTTP基本认证(Basic Authentication)实现

    JAVA中HTTP基本认证(Basic Authentication)实现

    HTTP 基本认证是一种简单的认证方法,本文主要介绍了JAVA中HTTP基本认证(Basic Authentication),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-07-07
  • 一文掌握SpringSecurity BCrypt密码加密和解密

    一文掌握SpringSecurity BCrypt密码加密和解密

    BCrypt就是一款加密工具,可以比较方便地实现数据的加密工作。也可以简单理解为它内部自己实现了随机加盐处理,这篇文章主要介绍了SpringSecurity BCrypt密码加密和解密,一文学会使用BCryptPasswordEncoder的方法,需要的朋友可以参考下
    2023-04-04
  • SpringBoot配置文件bootstrap和application区别及说明

    SpringBoot配置文件bootstrap和application区别及说明

    这篇文章主要介绍了SpringBoot配置文件bootstrap和application区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-06-06
  • classloader类加载器_基于java类的加载方式详解

    classloader类加载器_基于java类的加载方式详解

    下面小编就为大家带来一篇classloader类加载器_基于java类的加载方式详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • Mybatis-plus操作json字段实战教程

    Mybatis-plus操作json字段实战教程

    这篇文章主要介绍了Mybatis-plus操作json字段实战教程,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-02-02
  • Java如何处理数据成为树状结构

    Java如何处理数据成为树状结构

    这篇文章主要介绍了Java如何处理数据成为树状结构问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07

最新评论