你以为反射真的无所不能?至少JDK8以后很强大

胡老爷
• 阅读 2334

[TOC]

之前我们已经介绍了Java中框架常用的技术---反射。可以这么说反射方便了我们的开发。今天我们来说说他的短板,或者说我们今天在反射的基础上在进行方便化。

反射操作方法

在上一章节中我们学会了通过反射去调用方法。

public class App {
    public void test(String str, Integer integer) {
        System.out.println(str);
        System.out.println(integer);
    }
}

这个时候如果我想获取test方法对象的话应该这么做

Method testMethod = App.class.getMethod("test", String.class, Integer.class);

这里就不在赘述如何通过Method对象调用方法了。文章末尾会给出上一章节的地址。今天我们要研究的是Method如何获取方法参数这一块。看似简单却又是那么的传奇。我们看看下面一段代码执行的效果

public static void main(String[] args) throws ParseException, NoSuchMethodException {
        Method[] methods = App.class.getMethods();
        Method testMethod = App.class.getMethod("test", String.class, Integer.class);
        Class<?>[] parameterTypes = testMethod.getParameterTypes();
        Parameter[] parameters = testMethod.getParameters();
        for (Parameter parameter : parameters) {
            System.out.println(parameter.getName());
        }
    }

那么输出的两个参数名称是什么呢?一开始笔者这里想当然的认为是 str , name 。 相信此时的你应该和我一样认为是str , name 。

你以为反射真的无所不能?至少JDK8以后很强大

对的,你没看错返回的居然是无意义的名称 , arg0 , arg1.这就奇怪了。至于为什么呢?我现在还不想告诉你。下面会慢慢告诉你。

Spring的方法的优点

做过Javaweb开发的肯定都用过spring,springmvc , 在写controller层的时候我们都会在方法里直接写key值的名称,然后在请求地址中给相应的key赋值。

@RequestMapping(value = "/deptId", method = RequestMethod.GET)
public PagedResult<SysDept> selectSysDeptsByPK(Integer pageNumber, Integer pageSize) {
    return sysDeptService.selectSysDeptsByPK(deptId, pageNumber,  pageSize);
}

上述的controller我们在前端发送请求后会这样发送
http://{ip}:{port}/{projectName}/deptId?pageNumber=1&pageSize=5

这里我问一下你们有没有想过为什么springmvc它能够通过你传递的参数一一进行对应呢?我们上面已经尝试过通过反射是无法获取方法参数名称的。而springmvc无非就是反射操作方法的。这里是不是很神奇,不得不佩服springmvc的强大。强大到让人害怕。

反射如何实现Spring的方法

上面两个案例揭露了反射的缺点以及springmvc的强大。这里需要借助springmvc提供的一个工具ParameterNameDiscoverer 。 这个类顾名思义就是发现参数名称。在使用这个类之前我们先来了解下为什么反射获取不到方法名称。

这里需要简单说说Java执行过程,Java之所以可以跨容器是因为Java针对各个系统提供了不同jvm,所以我们开发前都需要安装不同版本的jdk,jdk里面提供了jvm,java 代码运行期间是通过jvm去操作class文件的。但是我们平时都是开发java文件的。所以在jvm执行之前会有一个编译期间。javac就是用来变异java文件为class文件的。

package com.zxhtom.test;

/**
 * Hello world!
 */
public class App {
    public void test(String str, Integer integer) {

    }
}

针对上述代码我们通过javac进行编译下试试看看效果。
javac App.java
编译完成之后会出现一个同名的class文件
你以为反射真的无所不能?至少JDK8以后很强大

然后我们在通过命令查看下这个class文件
javap -verbose App.class
你以为反射真的无所不能?至少JDK8以后很强大
通过查看App.java对应的字节码发现在javac编译的时候对于方法的名称根本不会去记录的。想想也对我执行方法的时候只需要按顺序将参数放进去就行了。根本不需要关心参数名称是什么。那么问题显而易见了jvm不需要参数名所以编译时过率了。但是我们反射想通过参数名称一一对应这样效率更快。那么是springmvc是如何解决的呢。

private static final ParameterNameDiscoverer parameterNameDiscoverer = new LocalVariableTableParameterNameDiscoverer();
public static void main(String[] args) throws ParseException, NoSuchMethodException {
    java.lang.reflect.Method testMethod = App.class.getMethod("test", String.class, Integer.class);
    String[] parameterNames = parameterNameDiscoverer.getParameterNames(testMethod);
    for (String parameterName : parameterNames) {
        System.out.println(parameterName);
    }
}

你以为反射真的无所不能?至少JDK8以后很强大
对,就是ParameterNameDiscoverer这个方法帮助了我们。这里简单说说ParameterNameDiscoverer作用。springmvc中会有一个默认的ParameterNameDiscoverer解释器DefaultParameterNameDiscoverer该类继承PrioritizedParameterNameDiscovererPrioritizedParameterNameDiscoverer这个类就是getParameterNames去获取方法名的。在springmvc中通过addDiscoverer方法有三个类注册到PrioritizedParameterNameDiscoverer
你以为反射真的无所不能?至少JDK8以后很强大

  • KotlinReflectionParameterNameDiscoverer : Spring5.0提供 ,但是也得jdk8及以上版本使用
  • StandardReflectionParameterNameDiscoverer : Spring4.0提供 ,但是也得jdk8及以上版本使用
  • LocalVariableTableParameterNameDiscoverer :Spring2.0就有了,对JDK版本没啥要求,完全Spring自己实现的获取字段名称,逻辑复杂些,效率稍微低一点

总结一下就是在springmvc4.0之前springmvc都是通过自己实现的一套代码去获取字节码然后分析的。这里能力有限就不分析了。
在4.0以后因为Java8的推出弥补了这个bug.springmvc也就都采用jdk提供的功能获取参数名了。下面我们来看看jdk8是如何解决这个问题的。

Java字节码

在上一节我们通过javac , javap命令进行了Java的编译了查看。我们发现class字节码中记录的信息有【常量区,类,方法】其中对于代码的记录有位置,堆,栈,行号等等。这也是我们jvm调优的依据。但是这仅仅是我们使用简单的javac的编译。

javac -g : 编译更加全面点
你以为反射真的无所不能?至少JDK8以后很强大
经过对比发现javac 和javac -g 的区别好像是javac -g 编译信息多出LocalVariableTable信息。

名称 解释
LineNumberTable 属性表存放方法的行号信息
LocalVariableTable   属性表中存放方法的局部变量信息

上图中通过javac -g 编译的信息中LocalVarableTable有三条数据,是因为在编译期间每个非静态方法第一个参数都是this.去除第一条剩下的其实就是我们需要的参数信息。但是我们这个时候去执行一下看看效果。
你以为反射真的无所不能?至少JDK8以后很强大

高级反射注意点

所谓的高级反射其实就是对jdk版本的要求,只要是jdk8的版本,就可以用jdk提供的parameter方法获取参数名了。在编译的时候需要加上 -parameters
你以为反射真的无所不能?至少JDK8以后很强大

javac的彩蛋

你以为反射真的无所不能?至少JDK8以后很强大

续点

 每日一笑

今天公司放假,和老婆商量好一起去她家,出发前夕,老婆说:给我家里人的礼物买好了么?然后我去卧室全搬出来了;给她说:这是你舅的,这是你叔的,这是你爷爷的,这是你奶奶的,这是你爸的,这是你妈的,这是…………然后我俩就打起来了!

上期答案

加入战队

<span id="addMe">加入战队</span>

微信公众号

你以为反射真的无所不能?至少JDK8以后很强大

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
4年前
java反射大全
作者对反射的理解:方法的调用(正常的调用:对象.方法()。反射调用方法:方法.对象())静态属性的调用(正常的调用:类.属性。反射调用:属性.类)常见反射的用法:        1.通过反射获取类Class<?demo1Class
Wesley13 Wesley13
4年前
java 成神之路 (二)
前一篇博客介绍了如何通过注解反射来实例化一个对象。前面提出的一个优化建议。上一版本,每个含有@DAL(http://my.oschina.net/daL)注解的变量都会通过反射去得到实例化对象。要是同一个工程里面每次都对同一个类通过反射实例化一个对象出来,这样太浪费资源了。现在我们就模拟spring的bean工厂,
Wesley13 Wesley13
4年前
java的反射机制
java中的反射可以将代码结构更加灵活,通过反射机制可以访问属性、方法和构造方法sun公司为我们提供的4大类反射:java.lang.reflect.methodjava.lang.Classjava.lang.reflect.modifierjava.lang.reflect.Constructor有以下几种方式:比如是Employ
浪人 浪人
4年前
Java基础与提高干货系列——Java反射机制
前言今天介绍下Java的反射机制,以前我们获取一个类的实例都是使用new一个实例出来。那样太low了,今天跟我一起来学习学习一种更加高大上的方式来实现。正文Java反射机制定义Java反射机制是指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以
红烧土豆泥 红烧土豆泥
4年前
创建型工厂设计模式之前置黑魔法(反射)
什么是反射?为什么说反射是黑魔法?为什么要在工厂设计模式前扯反射这东西?首先,既然在工厂设计模式前整它,肯定是在处理工厂设计模式时会用到它;既然都用黑魔法来形容它了,肯定是它异常强大;那什么是反射嘞?既然能被称为"反",那就肯定会有"正",那,正。。。是啥子嘞?想当然,既然平时我们都不怎么搞反射,那我们平日整的就是正了呗!平时我们实例化一个对象都是是
lzy lzy
4年前
RPC框架手撕之路---java反射以及动态代理机制
在上一篇文章中,我们提到了,RPC框架所需要的java基础,第一点就是java的动态代理机制,动态代理机制的基础是反射,无论是在实际编程或者是面试时,都是java知识的重中之重。java反射:定义:在运行状态中,对于任意一个类,都能够知道这一个类的所有属性和方法,对于任意一个对象都能够通过反射机制调用一个类的任意方法,这种动态获取类信息以及动态调用类方法
御弟哥哥 御弟哥哥
4年前
Java基础与提高干货系列 -- Java反射机制
前言今天介绍下Java的反射机制,以前我们获取一个类的实例都是使用new一个实例出来。那样太low了,今天跟我一起来学习学习一种更加高大上的方式来实现。正文Java反射机制定义Java反射机制是指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性
Wesley13 Wesley13
4年前
Java重点基础:反射机制
一、什么是反射?Java反射说的是在运行状态中,对于任何一个类,我们都能够知道这个类有哪些方法和属性。对于任何一个对象,我们都能够对它的方法和属性进行调用。我们把这种动态获取对象信息和调用对象方法的功能称之为反射机制。二、反射的三种方式
Wesley13 Wesley13
4年前
Java反射机制及适用场景
什么是Java反射机制?JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的以及动态调用对象的方法的功能称为Java的反射机制。反射的适用场景是什么?1.当你做一个软件可以安装插件的功能,你连插件的类型名称都不知道,你怎么实例化这个对象呢
Wesley13 Wesley13
4年前
Java反射例子汇总 Class Constructor Method Filed
一、反射概述  在平常的开发中Java的反射技术很少被用到,一般我们都是使用公司封装或者开源框架。而反射技术已经被包含到底层框架了,因此我们很少接触到。但是有些框架的原理或者源码如果想读懂就必须要理解并会使用反射技术。例如:EventBus、BufferKnife、android的插件化等等都会用到。理解了反射技术能够帮助我们更快的理解相关框架,也可以增
Wesley13 Wesley13
4年前
Java高级特性—反射和动态代理
1).反射  通过反射的方式可以获取class对象中的属性、方法、构造函数等,一下是实例:2).动态代理  使用场景:      在之前的代码调用阶段,我们用action调用service的方法实现业务即可。    由于之前在service中实现的业务可能不能够满足当先客户的要求,需要我们重新修改servic
胡老爷
胡老爷
Lv1
要相信,会有那么一个人会一直很爱自己。
文章
4
粉丝
0
获赞
0