Java程序员都要懂得知识点:反射
摘要:Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
反射就是把java类中的各种成分映射成一个个的Java对象。
例如:一个类有:成员变量、方法、构造方法、包等等信息,利用反射技术可以对一个类进行解剖,把个个组成部分映射成一个个对象。
(其实:一个类中这些成员方法、构造方法、在加入类中都有一个类来描述)
反射
Q: 调用类对象.class 和 forName(类名)的区别?
Class<A> classA = A.class;
Class<A> classA = Class.forName("A");
A: 仅使用.class不能进行第一次静态初始化, forname函数则可以
例如B是A的基类,下面这段代码如何?
假设有父子2个类,如下:
static class Parent { }
static class Son extends Parent{}
Q: 用instanceof 可以和父类比较吗,且会返回true吗?
Son son = new Son();
if (son instanceof Parent) {
System.out.println("a instanof B");
}
A: 可以比较,且返回true。
Q: 用getClass并用== 可以和父类比较吗,且会返回true吗,下面这样:
注意A是B的子类。
Son son = new Son();
if (son.getClass() == Parent.class){
System.out.println("son class == Parent.class");
}
A: 不可以,编译就会报错了。和Class<泛型>的 ==号比较有关。

因为getClass返回的是<? extends Son>, .class返回的是Class<Parent>
Q: 用getClass并用.equals可以和父类比较吗,且会返回true吗,下面这样:
Son son = new Son();
if (son.getClass().equals(Parent.class)){
System.out.println("son class.equals(Parent.class)");
}
A: 可以比较,正常编译, 但是会返回false,即不相等!
Q: getDeclaredXXX 有哪几种?
A: 5种:
- 注解Annotation
- 内部类Classed
- 构造方法Construcotor
- 字段Field
- 方法Method

Q:getMethods()返回哪些方法, getDeclaredMethods()会返回哪些方法?
A:
getMethods()返回 本类、父类、父接口 的public方法
getDeclaredMethods()只 返回本类的 所有 方法
其他getXXX和getDeclaredXXX的区别同理。
拿到Filed、Method、Constructor之后咋用
- Method可以invoke(object, args)
- Constructor可以newInstance(Object…)来做构造调用。
- Filed可以用get(object)、set(object)来设置属性值。
Q: 反射拿到Method对象后, 该对象.getModifiers() 是干嘛的?
A: 返回该方法的修饰符,并且是1个整数。

Q:
下面这段代码会发生什么?
package com.huawei.test
public class A {
public A(int i ) {
System.out.printf("i=" +i);
}
public static void main(String[] args) {
try {
A a = (A)Class.forName("com.huawei.test.A").newInstance();
} catch (ClassNotFoundException e) {
System.out.printf("ClassNotFoundException");
} catch (InstantiationException e) {
System.out.printf("InstantiationException");
} catch (IllegalAccessException e) {
System.out.printf("IllegalAccessException");
}
}
}
A:
打印InstantiationException初始化错误。因为A没有默认构造器了,所以不可以用newInstance来构造。应该改成这样,通过获取正确的构造器来进行构造。
A a = (A)Class.forName("A").getConstructor(int.class).newInstance(123);
Q:如何提高反射的效率?
A:
- 使用高性能反射包,例如ReflectASM
- 缓存反射的对象,避免每次都要重复去字节码中获取。(缓存!缓存!)
- method反射可设置method.setAccessible(true)来关闭安全检查。
- 尽量不要getMethods()后再遍历筛选,而直接用getMethod(methodName)来根据方法名获取方法
- 利用hotspot虚拟机中的反射优化技术(jit技术)
参考资料:
https://segmentfault.com/q/1010000003004720
https://www.cnblogs.com/coding-night/p/10772631.html
Q:
用反射获取到的method对象, 是返回一个method引用,还是返回1个拷贝的method对象?
A:
反射拿method对象时, 会做一次拷贝,而不是直接返回引用,因此最好对频繁使用的同一个method做缓存,而不是每次都去查找。

Q:
getMethods()后自己做遍历获取方法,和getMethod(methodName) 直接获取方法, 为什么性能会有差异?
A:
getMethods() 返回method数组时,每个method都做了一次拷贝。 getMethod(methodName)只会返回那个方法的拷贝, 性能的差异就体现在拷贝上。

Q:
获取方法时,jvm内部其实有缓存,但是返回给外部时依然会做拷贝。那么该method的缓存是持久存在的吗?
A:
不是持久存在的,内存不足时会被回收。源码如下:

private Class.ReflectionData<T> reflectionData() {
SoftReference<Class.ReflectionData<T>> reflectionData = this.reflectionData;
int classRedefinedCount = this.classRedefinedCount;
Class.ReflectionData rd;
return reflectionData != null && (rd = (Class.ReflectionData)reflectionData.get()) != null
&& rd.redefinedCount == classRedefinedCount ? rd : this.newReflectionData(reflectionData, classRedefinedCount);
}

可以看到这是一个软引用。
软引用的定义:内存紧张时可能会被回收,不过也可以通过-XX:SoftRefLRUPolicyMSPerMB参数控制回收的时机,只要发生GC就会将其回收。
如果reflectionData被回收之后,又执行了反射方法,那只能通过newReflectionData方法重新创建一个这样的对象了。
Q: 反射是线程安全的吗?
A:
是线程安全的。 获取反射的数据时,通过cas去获取。 cas概念可以见多线程一节。

Q:
a普通方法调用
b反射方法调用
c关闭安全检查的反射方法调用,性能差异如下:

b反射方法调用和c关闭安全检查的反射方法调用的性能差异在哪?普通方法调用和关闭安全检查的反射方法调用的性能差异在哪?
A:
- 安全检查的性能消耗在于
,SecurityManager.checkPermission(SecurityConstants.CHECK_MEMBER_ACCESS_PERMISSION); 这项检测需要运行时申请RuntimePermission(“accessDeclaredMembers”)。 所以如果不考虑安全检查, 对反射方法调用invoke时, 应当设置 Method#setAccessible(true) - 普通方法和反射方法的性能差异在于
- Method#invoke 方法会对参数做封装和解封操作
- 需要检查方法可见性
- 需要校验参数
- 反射方法难以内联
- JIT 无法优化
本文分享自华为云社区《java知识点问题精选之反射》,原文作者:breakDraw 。
Java程序员都要懂得知识点:反射的更多相关文章
- Java程序员需要学习的知识点
Java是全世界最受欢迎的3大编程语言之一,它可以开发出许多实用的WEB应用程序和桌面应用程序,更重要的一点,Java是跨平台的语言——编写一次,可以再任何地方运行.另外,Java也很容易入门,如果你 ...
- Java程序员都需要懂的「反射」
前言 只有光头才能变强. 文本已收录至我的GitHub精选文章,欢迎Star:https://github.com/ZhongFuCheng3y/3y 今天来简单写一下Java的反射.本来没打算写反射 ...
- Java程序员都应该去使用一下这款强大的国产工具类库
这不是标题党,今天给大家推荐一个很棒的国产工具类库:Hutool.可能有很多朋友已经知道这个类库了,甚至在已经在使用了,如果你还没有使用过,那不妨去尝试一下,我们项目组目前也在用这个.这篇文章来简单介 ...
- 2020年薪30W的Java程序员都要求熟悉JVM与性能调优!
前言 作为Java程序员,你有没有被JVM伤害过?面试的时候是否碰到过对JVM的灵魂拷问? 一.JVM 内存区域划分 1.程序计数器(线程私有) 程序计数器(Program Counter Reg ...
- 99.9%的Java程序员都说不清的问题:JVM中的对象内存布局?
本文转载自公众号:石彬的架构笔记,阅读大约需要8分钟. 作者:李瑞杰 目前就职于阿里巴巴,资深 JVM 研究人员 在 Java 程序中,我们拥有多种新建对象的方式.除了最为常见的 new 语句之外,我 ...
- 90% 的 Java 程序员都说不上来的为何 Java 代码越执行越快(1)- JIT编译优化
麻烦大家帮我投一票哈,谢谢 经常听到 Java 性能不如 C/C++ 的言论,也经常听说 Java 程序需要预热,那么其中主要原因是啥呢? 面试的时候谈到 JVM,也有很多面试官喜欢问,为啥 Java ...
- 90% 的 Java 程序员都说不上来的为何 Java 代码越执行越快(2)- TLAB预热
经常听到 Java 性能不如 C/C++ 的言论,也经常听说 Java 程序需要预热,那么其中主要原因是啥呢? 面试的时候谈到 JVM,也有很多面试官喜欢问,为啥 Java 程序越执行越快呢? 一般人 ...
- 阿里面试Java程序员都问些什么?
刚开始也是小白,也是一步步成成起来的.需要提的一点是,你将来是需要靠这个吃饭的,所以请对找工作保持十二分的热情,而且越早准备越好. 阿里一面 一面是在上午9点多接到支付宝的面试电话的,因为很期望能够尽 ...
- 看完这篇微服务架构设计思想,90%的Java程序员都收藏了
本博客强烈推荐: Java电子书高清PDF集合免费下载 https://www.cnblogs.com/yuxiang1/p/12099324.html 微服务 软件架构是一个包含各种组织的系统组织, ...
随机推荐
- vue watch route params change
vue watch route params change watch: { '$route.params.menuKey' (val, oldVal) { console.log('new rout ...
- Android 如何使用 WebView 加载 HTML 字符串和处理转译字符
Android 如何使用 WebView 加载 HTML 字符串和处理转译字符 css bug 在 WebView 中编译 Web 应用 如果您希望在客户端应用中提供 Web 应用(或只是网页),则可 ...
- CSS margin collapsing All In One
CSS margin collapsing All In One margin collapsing margin 塌陷 / margin 合并 会出现 margin collapsing 的 3 种 ...
- Chrome blocked third-party cookies
Chrome blocked third-party cookies Chrome Incognito Chrome 无痕模式 https://support.google.com/chrome/an ...
- html template tag
html template tag const tagName = `emoji-element`; const template = document.createElement('template ...
- HTML5 image rotate effect
HTML5 image rotate effect HTML5 实现旋转拨号键盘 简单的html中js实现图片中心旋转 https://download.csdn.net/download/q3168 ...
- React & redux-saga & effects & Generator function & React Hooks
React & redux-saga & effects & Generator function & React Hooks demos https://github ...
- 力扣832. 翻转图像-C语言实现-简单题
题目 传送门 文本 给定一个二进制矩阵 A,我们想先水平翻转图像,然后反转图像并返回结果. 水平翻转图片就是将图片的每一行都进行翻转,即逆序.例如,水平翻转 [1, 1, 0] 的结果是 [0, 1, ...
- 解决java POI导入Excel超时问题
由于要导入大量数据,后台会耗费很长时间,导致超时. 本项目前端request.js中设定的超时时间为150s. const service = axios.create({ baseURL: base ...
- 后端程序员之路 13、使用KNN进行数字识别
尝试一些用KNN来做数字识别,测试数据来自:MNIST handwritten digit database, Yann LeCun, Corinna Cortes and Chris Burgesh ...