本文分享自华为云社区《java知识点问题精选之反射》,原文作者:breakDraw 。

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.testpublic 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)
  • 普通方法和反射方法的性能差异在于
  1. Method#invoke 方法会对参数做封装和解封操作
  2. 需要检查方法可见性
  3. 需要校验参数
  4. 反射方法难以内联
  5. JIT 无法优化

点击关注,第一时间了解华为云新鲜技术~

Java程序员都要懂得知识点:反射相关推荐

  1. Java程序员都要懂得知识点:原始数据类型

    本文分享自华为云社区<Java知识点问题总结之原始数据类型>,原文作者:breakDraw. java原始数据类型有short.byte.int.long.boolean.char.flo ...

  2. Java程序员都是青春饭吗?

    Java程序员都是青春饭吗?过了35岁还能找到工作吗?会不会没人要了?随着IT行业的发展,互联网行业龙头企业中都流传一句话35岁是程序员的一个坎,过去了就没事,过不去就会被裁掉.正因为这句话大家才会认 ...

  3. 每个Java程序员都应该Follow的10个Twitter账号

    想让自己在第一时间了解Java世界正在发生什么吗? Twitter绝对是了解最新动态的优质资源.我列举了每个Java程序员都应该关注的10个twitter账号.在评论中可添加你喜欢的. 1 @java ...

  4. Java程序员都30岁了,还剩下5年“寿命”,这就是所谓的中年危机?

    Java程序员都30岁了,还剩下5年"寿命",这就是所谓的中年危机? 30岁时,我是一个程序员,离传说中的"退休"只有5年了,为了优雅从容的所谓"光荣 ...

  5. 一提到Reference 百分之九十九的java程序员都懵逼了

    原来的标题是:"一提到Reference 99.99%的java程序员都懵逼了",为啥改成汉字了呢?吐槽一下,因为CSDN出bug了,如果你用了%做标题,你的文章就别想用它的编辑器 ...

  6. 作为一名Java程序员,这些Spring知识点面试官常考

    点击蓝色"程序猿DD"关注我 回复"资源"获取独家整理的学习资料! 作者 | 丸纸 来源 | 极客时间 毋庸置疑,Spring 早已成为 Java 后端开发事实 ...

  7. 为什么很多Java程序员都转行做大数据了?

    如今大数据发展的越来越成熟.各大企业纷纷成立大数据部门.尤其BAT等一线互联网公司每天处理的数据量都是TB级别.大数据部门已成为这些企业的核心部门,数据已成为企业最核心的资产. 但是大数据人才缺口巨大 ...

  8. 优秀的Java程序员都在看哪些书?

    目录 一.立志存高远,笃行践初心 二.经典书籍 1.Java核心技术 2.Java编程思想 3.Java语言程序设计 4.Effective Java中文版(原书第3版) 5.Java并发编程实战 6 ...

  9. 无责任书评:每个Java程序员都应该深入理解Java虚拟机!

    Java这门语言的发展是很有意思的,它不像Python, Ruby 等完全是开源社区驱动,也不像C#,VB.NET主要由微软操刀.它是一个以Oracle(之前是Sun)为主,各大巨头一起参与,一起制定 ...

最新文章

  1. scrapy-redis源码抛析
  2. 时代亿信 文件共享访问控制网关
  3. [转载]Linux基础知识之挂载详解(mount,umount及开机自动挂载)
  4. 理论计算机科学研究生,清华大学理论计算机科学中心姚期智组招收保送研究生,请帮忙发到虎...
  5. AOP技术研究 再续
  6. linux中断申请之request_threaded_irq 【转】
  7. Linux下MySQL忘记密码
  8. 文字时钟罗盘动态html代码_抖音时钟原生JS文字钟源码下载
  9. 苹果App Store引发的悲喜狂欢
  10. 微信小程序image背景图片全屏显示(根据设备高度自适应背景图片)
  11. pip 安装包成功 但是import 失败
  12. 单元测试、API接口测试、灰盒测试
  13. python坐标定位手机元素_Appium+Python之元素定位和操作
  14. mining lorry和mining truck有什么不同。
  15. 渗透测试学习计划路线思维导图v1.0
  16. 语音识别芯片LD3320介绍续
  17. 通过图像相机进行mark定位,校正角度
  18. 软考笔记(一)高级系统架构师/分析师:计算机系统基础
  19. Typora自定义精美主题
  20. nutch 插件开发[资料整理]

热门文章

  1. 实践 + 理论 | API 接口安全性设计
  2. 继续!面试继续!Netty dubbo的通信方式
  3. Bootstrap 图片替换
  4. ROS笔记(15) Xacro
  5. ROS笔记(3) Melodic 的安装和配置
  6. IDEA中的将项目提交到远程git仓库
  7. vue开发 - 将方法绑定到window对象,给app端调用
  8. 微信小程序之换肤的功能
  9. python全栈开发基础【补充】包的补充
  10. Oracle PLSQL 客户端 连接Oracle12.2 出现权限问题的解决办法以及绿色版Oracle客户端的使用....