详解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 反射机制的资料请关注我们其它相关文章!

(0)

相关推荐

  • 在实践中了解Java反射机制应用

    引言 Java反射机制是一个非常强大的功能,在很多大型项目比如Spring, Mybatis都可以看见反射的身影.通过反射机制我们可以在运行期间获取对象的类型信息,利用这一特性我们可以实现工厂模式和代理模式等设计模式,同时也可以解决Java泛型擦除等令人苦恼的问题.本文我们就从实际应用的角度出发,来应用一下Java的反射机制. 反射基础 p.s: 本文需要读者对反射机制的API有一定程度的了解,如果之前没有接触过的话,建议先看一下官方文档的Quick Start. 在应用反射机制之前,首先我们先

  • java获取反射机制的3种方法总结

    反射机制的概念: 指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对于任意一个对象,都能调用它的任意一个方法.这种动态获取信息,以及动态调用对象方法的功能叫java语言的反射机制. 反射机制的应用: 生成动态代理,面向切片编程(在调用方法的前后各加栈帧). 反射机制的原理: 1 首先明确的概念: 一切皆对象----类也是对象. 2 然后知道类中的内容 :modifier constructor field method. 3 其次明白加载: 当Animal.class在硬盘中时

  • Java反射机制用法总结

    前言 本篇将从以下几个方面讲述反射的知识: class 的使用 方法的反射 构造函数的反射 成员变量的反射 一.什么是class类 在面向对象的世界里,万物皆对象.类是对象,类是java.lang.Class类的实例对象.另外class类只有java虚拟机才能new出来.任何一个类都是Class 类的实例对象.这实例对象有三种表达方式: public class User{ } public class ClassTest{ User u=new User(); //方式1: Class c1=

  • Java 反射机制原理与用法详解

    本文实例讲述了Java 反射机制原理与用法.分享给大家供大家参考,具体如下: 反射反射,程序员的快乐! 1.什么是反射? Java反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:并且能改变它的属性.而这也是Java被视为动态(或准动态,为啥要说是准动态,因为一般而言的动态语言定义是程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是

  • Java基于反射机制实现全部注解获取的方法示例

    本文实例讲述了Java基于反射机制实现全部注解获取的方法.分享给大家供大家参考,具体如下: 一 代码 class Info{ //给mytoString方法加了2个内建Annotation @Deprecated @SuppressWarnings(value = "This is a waring!") public String mytoString(){ return "hello world"; } } class GetAnnotations{ publi

  • 全面了解Java反射机制

    什么是反射 反射 (Reflection) 是Java的特征之一,它允许运行中的Java程序获取自身的信息,并且可以操作类或对象的内部属性. 通俗的来讲就是:通过反射机制,可以在运行时获得程序或程序集中每一个类型的成员和成员的信息. 注意这里的重点是:运行时,而不是编译时.我们常规情况下写的对象类型都是在编译期就确定下来的.而Java反射机制可以动态地创建对象并调用其属性,这样创建对象的方式便异常灵活了. 虽然通过反射可以动态的创建对象,增加了灵活性,但也不是什么地方都可用,还要考虑性能.编码量

  • 详解JAVA 反射机制

    什么是反射? 反射机制是在程序运行状态中,对于任意一个类,都能够获取这个类的所有属性和方法: 对于任意一个对象,都能够调用它的任意一个方法和属性: 这种动态获取信息以及动态调用对象的方法的功能称为java语言的反射机制. 反射的作用 1.可以实现简单的反编译,获取类中的属性和方法等基本信息,.class->java 2.通过反射机制获取类的属性.方法等 在使用eclipse时,通过对象引用.的方式,eclipse就会将这个对象中的所有属性和方法展示出来,这个就是利用的反射机制.其实反射应用最多的

  • 图文详解java反射机制及常用应用场景

    目录 一.什么是java反射? 二.HelloWorld 三.类加载与反射关系 四.操作反射的java类 4.1.获取Class对象的三种方法 4.2.获取Class类对象的基本信息 4.3.获得Class对象的成员变量 4.4.获取Class对象的方法 4.5.方法的调用 4.6.创建类的对象(实例化对象) 五.反射的常用场景 5.1.通过配置信息调用类的方法 5.2.结合注解实现特殊功能 5.3.按需加载jar包或class 六.反射的优缺点 一.什么是java反射? 在java的面向对象编

  • Java 类型信息详解和反射机制介绍

    RTTI RTTI(RunTime Type Information)运行时类型信息,能够在程序运行时发现和使用类型信息,把我们从只能在编译期知晓类型信息并操作的局限中解脱出来 传统的多态机制正是 RTTI 的基本使用:假设有一个基类 Shape 和它的三个子类 Circle.Square.Triangle,现在要把 Circle.Square.Triangle 对象放入 List<Shape> 中,在运行时,先把放入其中的所有对象都当作 Object 对象来处理,再自动将类型转换为 Shap

  • 详解JAVA类加载机制

    1.一段简单的代码 首先来一段代码,这个是单例模式,可能有的人不知道什么是单例模式,我就简单说一下 单例模式是指一个类有且只有一种对象实例.这里用的是饿汉式,还有懒汉式,双检锁等等.... 写这个是为了给大家看一个现象 class SingleTon{ public static int count1; public static int count2=0; private static SingleTon instance=new SingleTon(); public SingleTon()

  • 详解Java 反射和反射的应用场景

    反射机制介绍 JAVA 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为 java 语言的反射机制. 获取 Class 对象的两种方式 如果我们动态获取到这些信息,我们需要依靠 Class 对象.Class 类对象将一个类的方法.变量等信息告诉运行的程序.Java 提供了两种方式获取 Class 对象: 1.知道具体类的情况下可以使用: Class alunbarCla

  • 详解Java序列化机制

    概况 在程序中为了能直接以 Java 对象的形式进行保存,然后再重新得到该 Java 对象,这就需要序列化能力.序列化其实可以看成是一种机制,按照一定的格式将 Java 对象的某状态转成介质可接受的形式,以方便存储或传输.其实想想就大致清楚基本流程,序列化时将 Java 对象相关的类信息.属性及属性值等等保存起来,反序列化时再根据这些信息构建出 Java 对象.而过程可能涉及到其他对象的引用,所以这里引用的对象的相关信息也要参与序列化. Java 中进行序列化操作需要实现 Serializabl

  • 详解Java反射创建对象

    一.什么是反射 Java Reflaction in Action中的解释:反射是运行中的程序检查自己和软件运行环境的能力,它可以根据它发现的进行改变.通俗的讲就是反射可以在运行时根据指定的类名获得类的信息 个人理解:就是我们对于创建对象我们除了通过 new关键字创建外,还能通过什么创建呢?private的属属性真的不能获取吗?反射就能做到打破这些所谓的规则反射和new创建对象谁的效率高? new 二.通过类对象调用newInstance()方法,适用于无参构造方法 2.1 类名.class p

  • 详解JAVA类加载机制(推荐)

    JAVA源码编译由三个过程组成: 1.源码编译机制. 2.类加载机制 3.类执行机制 我们这里主要介绍编译和类加载这两种机制. 一.源码编译 代码编译由JAVA源码编译器来完成.主要是将源码编译成字节码文件(class文件).字节码文件格式主要分为两部分:常量池和方法字节码. 二.类加载 类的生命周期是从被加载到虚拟机内存中开始,到卸载出内存结束.过程共有七个阶段,其中到初始化之前的都是属于类加载的部分 加载----验证----准备----解析-----初始化----使用-----卸载 系统可能

  • 详解Java反射各种应用

    Java除了给我们提供在编译期得到类的各种信息之外,还通过反射让我们可以在运行期间得到类的各种信息.通过反射获取类的信息,得到类的信息之后,就可以获取以下相关内容: Class对象 构造器 变量 方法 私有变量与私有方法 注解 泛型 数组 本文也将从上面几个方面来介绍Java反射.本文涉及的所有代码均在反射代码 首先放出一个Java类作为反射的研究对象,类的内容如下: public abstract class FatherObject implements Runnable{ public v

  • 详解JAVA SPI机制和使用方法

    JAVA SPI 简介 SPI 是 Java 提供的一种服务加载方式,全名为 Service Provider Interface.根据 Java 的 SPI 规范,我们可以定义一个服务接口,具体的实现由对应的实现者去提供,即服务提供者.然后在使用的时候再根据 SPI 的规范去获取对应的服务提供者的服务实现.通过 SPI 服务加载机制进行服务的注册和发现,可以有效的避免在代码中将具体的服务提供者写死.从而可以基于接口编程,实现模块间的解耦. SPI 机制的约定 1 在 META-INF/serv

随机推荐