关于Java中反射机制的深入讲解

前言

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

一:Class类

在面向对象的世界里,万物皆对象。类也是对象,类是java.lang.Class类的实例对象。

Class类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。

基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。

Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。

上面来自于JDK的罗里吧嗦,下面我来说下自己的体会:

类不是抽象的,类是具体的!

类是.class字节码文件,要想获取一个Class实例对象,首先需要获取.class字节码文件!

然后调用Class对象的一些方法,进行动态获取信息以及动态调用对象方法!

二:类类型

新建一个Foo类。Foo这个类也是实例对象,是Class的实例对象。

不知道你是否在意过类的声明与方法的声明:

public class Foo{
 Foo(){
  //构造方法
 }
}
public Foo method(){
 //...
}

我们知道public后跟返回类型,也就可以知道class也是一个类型。

如何表示Class的实例对象?

 public static void main(String[] args) {
 //Foo的实例对象,new 就出来了
 Foo foo1 = new Foo();

 //如何表示?
 //第一种:告诉我们任何一个类都有一个隐含的静态成员变量class
 Class c1 = Foo.class;

 //第二种:已经知道该类的对象通过getClass方法
 Class c2 = foo1.getClass();
 System.out.println(c1 == c2);

 //第三种:动态加载
 Class c3 = null;
 try {
  c3 = Class.forName("cn.zyzpp.reflect.Foo");
 } catch (ClassNotFoundException e) {
  e.printStackTrace();
 }

 System.out.println(c2 == c3);
 }

上述打印结果全是true

尽管 c1或c2 都代表了Foo的类类型,一个类只能是Class类的一个实例变量。

我们完全可以通过类的类类型(Class类型)创建类的实例对象。

 //此时c1 c2 c3为Class的实例对象
 try {
//  Foo foo = (Foo)c1.newInstance();
  Foo foo = (Foo)c3.newInstance();
  foo.print();
 } catch (InstantiationException e) {
  e.printStackTrace();
 } catch (IllegalAccessException e) {
  e.printStackTrace();
 }

静态加载

new 创建对象是静态加载类,在编译时刻就需要加载所有的可能使用到的类 。

动态加载

使用 Class.forName("类的全称") 加载类称作为动态加载 。

编译时刻加载类是静态加载类,运行时刻加载类是动态加载类。

举个例子

定义Office类

public class Office {
 public void print() {
 System.out.println("office");
 }
}

定义Loading类

public class Loading {

 public static void main(String[] args) {
 try {
  //在运行时再动态加载类
  //arg[0] 为java执行命令时传的参数
  Class<?> a = Class.forName(args[0]);
  Office office = (Office) a.newInstance();
  office.print();
 } catch (ClassNotFoundException e) {
  e.printStackTrace();
 } catch (IllegalAccessException e) {
  e.printStackTrace();
 } catch (InstantiationException e) {
  e.printStackTrace();
 }
 }
}

执行过程

D:\>javac -encoding utf-8 Loading.java Office.java

D:\>java Loading Office
office

通过Class a=Class.forName(arg[0])动态加载获取类,因编译时不知道使用哪个类,因此编译没有加载任何类,直接通过编译,运行时,根据 java Loading Office (office是一个类类型/类,下标arg[0]),去确定a是哪个类。这就是动态加载。如果Office类不存在,此时运行会报错。这就是为何有时候会出现编译通过,运行报错的原因。

动态加载一个好处,就是可以随时增加需要编译的类。例如把Office改造为抽象类或接口,定义不同的子类,动态选择加载。

三:类的反射

通过上面的三种方法获取到类的类类型,就可以获取到该类的成员方法,成员变量,方法参数注释等信息。

方法对象是Method类,一个成员方法就是一个Method对象。

方法 解释
getMethods() 返回该类继承以及自身声明的所有public的方法数组
getDeclaredMethods() 返回该类自身声明的所有public的方法数组,不包括继承而来

成员变量也是对象,是java.lang.reflect.Field对象,Field类封装了关于成员变量的操作。

方法 解释
getFields() 获取所有的public的成员变量信息,包括继承的。
getDeclaredFields() 获取该类自己声明的成员变量信息,public,private等

获取Java语言修饰符(public、private、final、static)的int返回值,再调用Modifier.toString()获取修饰符的字符串形式,注意该方法会返回所有修饰符。

方法 解释
getModifiers() 以整数形式返回由此对象表示的字段的 Java 语言修饰符。

获取注释

方法 解释
getAnnotations() 返回此元素上存在的所有注释。
getDeclaredAnnotations() 返回直接存在于此元素上的所有注释。

构造函数也是对象,是java.lang.reflect.Constructor的对象。

方法 解释
getConstructors() 返回所有public构造方法
getDeclaredConstructors() 返回类的所有构造方法,不止public

完整示例

 private void printClassMessage(Object obj){
 //要获取类的信息,首先获取类的类类型
 Class clazz = obj.getClass();
 //获取类的名称
 System.out.println(Modifier.toString(clazz.getModifiers())+" "+ clazz.getClass().getName()+" "+clazz.getName()+"{");

 System.out.println("----构造方法----");

 //构造方法
 Constructor[] constructors = clazz.getDeclaredConstructors();
 for (Constructor constructor: constructors){
  //构造方法修饰符与名字
  System.out.print(Modifier.toString(constructor.getModifiers())+" "+constructor.getName()+"(");
  //构造函数的所有参数类型
  Class[] parameterTypes = constructor.getParameterTypes();
  for (Class c: parameterTypes){
  System.out.print(c.getName()+", ");
  }
  System.out.println("){}");
 }

 System.out.println("----成员变量----");

 //成员变量
 Field[] fields = clazz.getDeclaredFields();
 for (Field field: fields){
  System.out.println(" "+Modifier.toString(field.getModifiers())+" "+field.getType().getName() + " " + field.getName()+";");
 }

 System.out.println("----成员方法----");

 //Method类,方法对象,一个成员方法就是一个Method对象
 Method[] methods = clazz.getDeclaredMethods();
 for (Method method : methods){
  //获取方法返回类型
  Class returnType = method.getReturnType();
  //获取方法上的所有注释
  Annotation[] annotations = method.getAnnotations();
  for (Annotation annotation: annotations){
  //打印注释类型
  System.out.println(" @"+annotation.annotationType().getName()+" ");
  }
  //打印方法声明
  System.out.print(" "+Modifier.toString(returnType.getModifiers())+" "+returnType.getName()+" "+method.getName()+"(");
  //获取方法的所有参数类型
  Class<?>[] parameterTypes = method.getParameterTypes();
  //获取方法的所有参数
  Parameter[] parameters = method.getParameters();
  for (Parameter parameter: parameters){
  //参数的类型,形参(全是arg123..)
  System.out.print(parameter.getType().getName()+" "+parameter.getName()+", ");
  }
  System.out.println(")");
 }
 System.out.println("}");
 }

以String对象为例,打印结果:

public final java.lang.Class java.lang.String{
----构造方法----
public java.lang.String([B, int, int, ){}
 java.lang.String([C, boolean, ){}
----成员变量----
 private final [C value;
 private int hash;
----成员方法----
 @java.lang.Deprecated
 public abstract final void getBytes(int arg0, int arg1, [B arg2, int arg3, )
 ......
}

四:方法的反射

定义了一个类Foo用于测试

public class Foo{
 public void print(String name,int num) {
 System.out.println("I am "+name+" age "+num);
 }
}

目标:通过反射获取该方法,传入参数,执行该方法!

1.获取类的方法就是获取类的信息,获取类的信息首先要获取类的类类型

Class clazz = Foo.class;

2.通过名称+参数类型获取方法对象

Method method = clazz.getMethod("print", new Class[]{String.class,int.class});

3.方法的反射操作是通过方法对象来调用该方法,达到和new Foo().print()一样的效果

方法若无返回值则返回null

Object o = method.invoke(new Foo(),new Object[]{"name",20});

五:通过反射认识泛型

 public static void main(String[] args) {
  ArrayList<String> stringArrayList = new ArrayList<>();
  stringArrayList.add("hello");
  ArrayList arrayList = new ArrayList();

  Class c1 = stringArrayList.getClass();
  Class c2 = arrayList.getClass();

  System.out.println(c1 == c2);

 }

打印结果为true

c1==c2的结果返回说明编译之后集合的泛型是去泛型化的。换句话说,泛型不同,对类型没有影响。

Java中集合的泛型其实只是为了防止错误输入,只在编译阶段有效,绕过编译就无效。

验证

我们可以通过反射来操作,绕过编译。

 public static void main(String[] args) {
  ArrayList<String> stringArrayList = new ArrayList<>();
  stringArrayList.add("hello");
  ArrayList arrayList = new ArrayList();
  Class c1 = stringArrayList.getClass();
  Class c2 = arrayList.getClass();
  System.out.println(c1 == c2);

  try {
   Method method = c1.getMethod("add",Object.class);
   method.invoke(stringArrayList,20);
   System.out.println(stringArrayList.toString());
  } catch (NoSuchMethodException e) {
   e.printStackTrace();
  } catch (IllegalAccessException e) {
   e.printStackTrace();
  } catch (InvocationTargetException e) {
   e.printStackTrace();
  }

 }

打印结果:

true
[hello, 20]

成功绕过了泛型<String>的约束。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Java反射机制实例代码分享

    本文旨在对Java反射机制有一个全面的介绍,希望通过本文,大家会对Java反射的相关内容有一个全面的了解. 阅读本文之前,大家可先行参阅<重新理解Java泛型>. 前言 Java反射机制是一个非常强大的功能,在很多大型项目比如Spring, Mybatis都可以看见反射的身影.通过反射机制我们可以在运行期间获取对象的类型信息,利用这一特性我们可以实现工厂模式和代理模式等设计模式,同时也可以解决Java泛型擦除等令人苦恼的问题.本文我们就从实际应用的角度出发,来应用一下Java的反射机制. 反射

  • Java高级特性之反射机制实例详解

    本文实例讲述了Java高级特性之反射机制.分享给大家供大家参考,具体如下: 老规矩我们还是先提出几个问题,一门技术必然要能解决一定的问题,才有去学习掌握它的价值 一. 什么是反射? 二.反射能做什么? 一. 什么是反射? 用在Java身上指的是我们可以于运行时加载.探知.使用编译期间完全未知的classes.换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体.或对其fields设值.或唤起其methods. 如果你是一个

  • Java反射机制的精髓讲解

    1,什么是反射? java的反射,允许程序在运行时,创建一个对象,获取一个类的所有相关信息等. 2,Class类 要了解反射,就绕不开Class类. 我们平时开发的类,例如ClassA,一般会有一些属性,会有几个构造方法,也会有一些普通方法,我们还可以使用ClassA来创建对象,例如ClassA classA = new ClassA(). java程序在运行时,其实是很多类的很多个对象之间的协作.jvm如何管理这些类呢?它如何知道各个类的名称,每个类都有哪些属性和哪些方法呢? jvm会给每个类

  • Java编程反射机制用法入门与实例总结

    本文实例讲述了Java编程反射机制用法.分享给大家供大家参考,具体如下: 前言:反射:动态获取类 (字节码文件 如本篇中的Person.class),并对其成员进行运行.反射在Android应用层的开发中可能遇到会稍微少一点,但对于想打通底层的小伙伴来说,必须要熟练掌握运用. 实体类 Person.java package com.sunwenou.reflect;// 包名 public class Person { private String name; private int age;

  • JAVA反射机制中getClass和class对比分析

    java有两个获得类名的方法getClass()和class(),这两个方法看似一样,实则不然.这两个方法涉及到了java中的反射. 所谓反射,可以理解为在运行时期获取对象类型信息的操作.传统的编程方法要求程序员在编译阶段决定使用的类型,但是在反射的帮助下,编程人员可以动态获取这些信息,从而编写更加具有可移植性的代码.严格地说,反射并非编程语言的特性,因为在任何一种语言都可以实现反射机制,但是如果编程语言本身支持反射,那么反射的实现就会方便很多. 类型类 要知道类型信息在运行时是如何表示的,这是

  • 浅析Java 反射机制的用途和缺点

    反射的用途 Uses of Reflection Reflection is commonly used by programs which require the ability to examine or modify the runtime behavior of applications running in the Java virtual machine. This is a relatively advanced feature and should be used only by

  • 实例讲解Java中动态代理和反射机制

    反射机制 Java语言提供的一种基础功能,通过反射,我们可以操作这个类或对象,比如获取这个类中的方法.属性和构造方法等. 动态代理:分为JDK动态代理.cglib动态代理(spring中的动态代理). 静态代理 预先(编译期间)确定了代理者与被代理者之间的关系,也就是说,若代理类在程序运行前就已经存在了,这种情况就叫静态代理 动态代理 代理类在程序运行时创建的代理方式.也就是说,代理类并不是在Java代码中定义的,而是在运行期间根据我们在Java代码中的"指示"动态生成的. 动态代理比

  • 关于Java中反射机制的深入讲解

    前言 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制. 一:Class类 在面向对象的世界里,万物皆对象.类也是对象,类是java.lang.Class类的实例对象. Class类的实例表示正在运行的 Java 应用程序中的类和接口.枚举是一种类,注释是一种接口.每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共

  • JAVA中反射机制和模块化的深入讲解

    目录 一.类加载 1.1类加载描述 1.2类的加载 1.3类的连接 1.4类的初始化 1.4.1类初始化的作用 1.4.2初始化步骤 1.4.3类的初始化时机 二.反射 2.1反射的概述 2.2获取Class类对象的三种方式 2.3反射获取构造方法 2.4反射获取成员变量 2.5反射获取成员方法 2.6反射的案例 2.6.1反射练习之越过泛型检查 2.6.2运行配置文件中指定类的指定方法 三.模块化 3.1模块化概述 ​​​​​​​​​​​​​​3.2模块化使用 3.3模块化的基本使用 总结 一

  • 详解java中反射机制(含数组参数)

    详解java中反射机制(含数组参数) java的反射是我一直非常喜欢的地方,因为有了这个,可以让程序的灵活性大大的增加,同时通用性也提高了很多.反射原理什么的,我就不想做过大介绍了,网上一搜,就一大把.(下面我是只附录介绍下) Reflection 是Java被视为动态(或准动态)语言的一个关键性质.这个机制允许程序在运行时透过Reflection APIs取得任何一个已知名称的class的内部信息,包括其modifiers(诸如public, static 等等).superclass(例如O

  • Java中反射机制和作用详解

    前言 很多刚学Java反射的同学可能对反射技术一头雾水,为什么要学习反射,学习反射有什么作用,不用反射,通过new也能创建用户对象. 那么接下来大师就带你们了解一下反射是什么,为什么要学习反射? 下面我们首先通过一个实例来说明反射的好处: 方法1.不用反射技术,创建用户对象,调用sayHello方法 1.1 我们首先创建一个User类 package com.dashi; /** * Author:Java大师 * User对象,包含用户的id和姓名以及sayHello方法 */ public

  • Java反射机制的简单讲解

    🌱1. 什么是反射机制? 首先大家应该先了解两个概念,编译期和运行期,编译期就是编译器帮你把源代码翻译成机器能识别的代码,比如编译器把java代码编译成jvm识别的字节码文件,而运行期指的是将可执行文件交给操作系统去执行,JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制 简单说,反射机制值得是程序在运行时能够获取自身的信息.在java中,只要给定类的名字,那么就可以通过反射机制来获得类的所有信息. 🌱2. java反射机制提供了什么功能? 在运行时能够判断任意一个对象所属的类 在运行时构造任意一个类的对象 在运行时判断任意一个类所具有的成员变量和方法 在运行时调用任一对象的方法 在运行时创建新类对象 🌱3.new和反射创建有什么区别呢? ne

  • Java的反射机制---动态调用对象的简单方法

    唉!我还真是在面试中学习新东东啊,一个公司刚刚给了个测试,不过我很奇怪的是为什么web developer的职位居然考java的反射机制题,不过学习研究一下反射机制对我来说是件好事啦! 先说说什么是java反射机制吧,在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这 种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制.主要功能:在运行时判断任意一个对象所属的类:在运行时构造任意一个类的对 象:在运行时判断任意一个

  • 简单总结Java的反射机制的运用

    Java 的反射机制是使其具有动态特性的非常关键的一种机制,也是在JavaBean 中广泛应用的一种特性. 简单来说,一个类或者一个对象是拥有下面几种属性的: Method,Constructor,Field,其大致结构类图如下: 我们现在用代码来说明问题: 首先,我们看Class类,在Class类中,我们可以看见下面的几个重要的方法: getInterfaces() getSuperClass(); isInterface(); 这是用来得到一个类的接口或者超类,以及判断这个类是不是一个接口:

  • java中注解机制及其原理的详解

    java中注解机制及其原理的详解 什么是注解 注解也叫元数据,例如我们常见的@Override和@Deprecated,注解是JDK1.5版本开始引入的一个特性,用于对代码进行说明,可以对包.类.接口.字段.方法参数.局部变量等进行注解.它主要的作用有以下四方面: 生成文档,通过代码里标识的元数据生成javadoc文档. 编译检查,通过代码里标识的元数据让编译器在编译期间进行检查验证. 编译时动态处理,编译时通过代码里标识的元数据动态处理,例如动态生成代码. 运行时动态处理,运行时通过代码里标识

  • Java基础--反射机制

    反射 反射可以使我们很方便的创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代码链接.反射允许我们在编写和执行时,使我们的代码能够接入装载到JVM中的类的内部信息,而不是源代码中选定的类协作的代码. 反射机制作用 反编译:.class -> .java 通过反射机制访问Java对象的属性,方法,构造方法 反射的使用 反射机制获取类的三种方式 Class c1 = Class.forName("com.webb.basis.reflect.Demo"); // 一般采

随机推荐