vector父类类型可以存放子类吗_拼夕夕三轮面经:被问到反射和泛型的bug,你踏空了吗?...

  点击上方“JavaEdge”,关注公众号

设为“星标”,好文章不错过!386534a9f0127c8872590db31646fbab.png

1 当反射遇见方法重载

aa99579d2ce06b555432ddbad60b4366.png

重载grade方法,入参分别为int、Integer。64fbc5782e4d44a7c8ff3e86726d15df.png若不通过反射这种高级编程方式,选用哪个重载方法自然很清晰,比如传666走int参数重载方法,传入Integer.valueOf(“666”)走Integer重载。

但你若墨守成规认为反射调用方法也是根据入参类型确定方法重载,那就掉坑了。
使用getDeclaredMethod获取 grade方法,然后传入Integer.valueOf(“36”)567abd5ee2f51c0c9bc4b03731cd4748.png5d32e0dfa70d7f3d37d90c6c83e82484.png

因为通过反射进行方法调用首先是

通过方法签名来确定方法

cb3443b97042b684e3435a59422d1263.png

本例的getDeclaredMethod传入的参数类型Integer.TYPE其实一直代表int。4584e7dc8b6f7826abae64f96d604fd4.png

所以实际执行方法时传包装类型、基本类型,最终都是调用int入参的grade方法。

f5c9a740aedb965b58ce3d816d3a1587.png

修正方案

a897eccc3ace704e7c02a44bab06a5eb.png

Integer.TYPE改为Integer.class,实际执行的参数类型就是Integer了。且无论传包装类型/基本类型,最终都会调用Integer为入参的grade方法。

所以反射调用方法,是以反射获取方法时传入的方法名和参数类型来确定调用的方法。

386534a9f0127c8872590db31646fbab.png

2 当泛型因类型擦除遇见桥接方法

aa99579d2ce06b555432ddbad60b4366.png

泛型作为一种编程范式,使得开发者可以使用类型参数替代精确类型,实例化时再指明具体类型。也利于代码重用,将一套代码应用到多种数据类型

泛型的类型检测,可以在编译时暴露大多数泛型编码错误。但由于历史兼容性而妥协的泛型类型擦除,在运行时才会暴露很多坑。

案例

cb3443b97042b684e3435a59422d1263.png

期望在类字段内容变动时记录日志,于是开发同学就想到定义一个泛型父类,并在父类中定义一个统一的日志记录方法,子类可继承该方法。上线后总出现日志重复记录问题。

父类9eb906565910b645439023ed07703105.png

子类Child1 未提供父类泛型参数且定义了一个参数为String而非TsetValue。期望覆盖父类的setValue实现。318c76fd2bf0deffe794f30d38ee557f.png

子类方法的调用是通过反射。83a98fc58582200e5c1e84a63cf1c2bd.png

虽Parent的value字段正确设置JavaEdge,但父类setValue调用了两次,计数器而显示2503293ca45968ffe003d49e51c17639c.png两次Parent的setValue方法调用,是因为getMethods找到了两个setValue的,分属于父类/子类。

f5c9a740aedb965b58ce3d816d3a1587.png

子类重写父类方法失败原因

a897eccc3ace704e7c02a44bab06a5eb.png
  • 子类未指定String泛型参数,父类的泛型方法setValue(T value)泛型擦除后是setValue(Object value),于是子类入参String的setValue被当作了新方法

  • 子类的setValue方法未加@Override注解,编译器未能检测到重写失败。

重写子类方法时,务必使用@Override注解。

但有人认为问题是反射API使用不当而未意识到重写失败。查文档后才发现

  • getMethods能获得当前类和父类的所有public方法

  • getDeclaredMethods仅获得当前类所有的public、protected、package和private方法

于是用getDeclaredMethods替换getMethodseb995a83671f4a6e473a1ef97e90e545.png

这虽能解决重复记录日志,但未解决子类重写父类方法失败,日志:da5a6c57041aed435f4b6d1ae8c72670.png

当其他人使用Child1时还是会发现有俩setValue,让人困惑。

重新实现Child2,继承Parent时String作为泛型T类型,并使用@Override注解setValue,实现有效的方法重写e7dc9488cbf40146fb7aec5323e484c4.png

还是出现重复日志acc0412e20ecd33ce3710108a6bbd699.png

Child2的setValue调了两次。难道是JDK的反射出Bug了!
通过getDeclaredMethods查找到的方法肯定来自Child2本身;而且Child2类中看起来也只有一个setValue,怎么可能还重复?

调试发现,Child2类其实有俩setValue:入参分别是String/Object。65beb4ffb9ef988d169190ed9156980c.png56ff5bd7390fc73f0a6272be7a560d9e.png

这就是泛型类型擦除导致。

解密反射下的泛型擦除天坑

cb3443b97042b684e3435a59422d1263.png

Java泛型类型在编译后被擦除为Object。子类虽指定父类泛型T类型是String,但编译后T会被擦除成为Object,所以父类setValue入参是Object,value也是Object。
若Child2 setValue想覆盖父类,那入参也须为Object。所以,编译器会为我们生成一个桥接方法

Child2类的class字节码:

01816e8f76bbb0607bf3992bca475d8f.png

若编译器未帮我们实现该桥接方法,那Child2重写的是父类泛型类型擦除后、入参是Object的setValue。这俩方法参数,一个String一个Object,明显不符合Java语义:

class Parent {
AtomicInteger updateCount = new AtomicInteger();private Object value;public void setValue(Object value) {
System.out.println("调用 Parent 的 setValue");this.value = value;
updateCount.incrementAndGet();}}class Child2 extends Parent {@Overridepublic void setValue(String value) {
System.out.println("调用 Child2 的 setValue");super.setValue(value);}}

验证:使用jclasslib打开Child2,可看到入参为Object的桥接方法上标记public synthetic bridge。synthetic代表由编译器生成的不可见代码,bridge代表这是泛型类型擦除后生成的桥接代码c7f0ddfc2027eb7868dd7d6b2c10b671.png

修正方案

cb3443b97042b684e3435a59422d1263.png

使用method的isBridge方法,来判断方法是不是桥接方法:

  • 通过getDeclaredMethods方法获取到所有方法后,必须同时根据方法名setValue和非isBridge两个条件过滤,才能实现唯一过滤

  • 使用Stream时,如果希望只匹配0或1项的话,可以考虑配合ifPresent来使用findFirst方法。

75f4b58b3b3dc5b5a618b1647ee22ae5.png

使用反射查询类方法清单时:

  • getMethods和getDeclaredMethods是有区别的,前者可以查询到父类方法,后者只能查询到当前类

  • 反射进行方法调用要注意过滤桥接方法。

往期推荐

大厂如何解决数值精度/舍入/溢出问题

大厂数据库事务实践-事务生效就能保证正确回滚?

线上问题事迹(一)数据库事务居然都没生效?

硬核干货:HTTP超时、重复请求必见坑点及解决方案

给大忙人们看的Java NIO教程之Channel

bfa6f9c68d730c66d8822ce4b476e3d1.gif

目前交流群已有 800+人,旨在促进技术交流,可关注公众号添加笔者微信邀请进群

f97b81055ecd40a898604d9b6d02cf99.png

喜欢文章,点个“在看、点赞、分享”素质三连支持一下~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/503636.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

雨林木风系统封装工具封装xp_如何用小丸工具大幅度压缩视频且画质损失较小?...

如何大幅度把视频体积压下去,并保持清晰度,就只有用比视频原本低的码率,降低分辨率和压缩音频这三种方法。降低码率可以使用CRF来控制,值越大码率越小,成反比。一些录屏和直播的视频产生的码率都很高,文件体…

java 变量作用域 c语言_C语言深入理解 - 常量与变量

《C语言深入理解系列 - 常量与变量》查看其它博文请关注原创作者。本文系本站原创,欢迎转载! 转载请注明出处:常量与变量正所谓静中有动,动中有静,常量与变量亦是如此,它们之前相互依赖,相互影响。关于常量与变量,很多…

数据可视化demo_为更快读懂报表,我们将数据可视化了

将数据可视化后,即使是复杂难懂的数据也会瞬间变得简单易懂,难就难在怎么快速将海量数据中的复杂数据信息提取,不同场景下该用那种方式展示数据更直观易懂。而这些,在奥威BI数据可视化软件上都有了答案。在常用图表的基础上&#…

swift for循环_Swift | 实战一个简单的素数计算器demo

Swift实战一个简单的素数计算器demo本期我们来介绍如何用storyboard来实现一个素数计算器demo,storyboard可以明确地知道界面上的组件与代码的关系,而且比起仅用代码写要方便不少。No.1制作一个简单的界面制作一个简单的界面我们首先要使用storyboard来绘…

java文件传输连接方式_Java 学习笔记 网络编程 使用Socket传输文件 CS模式

Socket的简单认识Socket是一种面向连接的通信协议,Socket应用程序是一种C/S(Client端/Server端)结构的应用程序 Socket是两台机器间通信的端点。 Socket是连接运行在网络上的两个程序间的双向通讯端点。Socket通信原理Server服务端的输入流相当于Client客户端的输出…

创建线程的三种方法_Netty源码分析系列之NioEventLoop的创建与启动

前言前三篇文章分别分析了 Netty 服务端 channel 的初始化、注册以及绑定过程的源码,理论上这篇文章应该开始分析新连接接入过程的源码了,但是在看源码的过程中,发现有一个非常重要的组件:NioEventLoop,出现得非常频繁…

java正则表达式匹配xml标签_用正则表达式匹配HTML\XML等文件中的标签

测试用HTML源文件:View Code《完美世界:天界的召唤》缤纷圣诞总动员[ 中华网 1小时前]经过了平安夜和圣诞节,节日的气氛被推到了最高点!《完美世界:天界的召唤》为玩家准备了精彩纷呈的圣诞节活动,而玩家也…

java spring hiberate_Java程序员:Spring Boot和Hibernate一起使用的技巧

Hibernate不需要多介绍,它是Java中最受欢迎的ORM。同样,Spring Boot是功能最强大且易于使用的框架。本文并不是描述一些关于Hibernate或Spring Boot的用法,因为有很多。相反,我们将研究同时使用它们时可能遇到的一些常见错误以及如…

postgresql 查询序列_RazorSQL for Mac(数据库工具查询) v9.0.9

RazorSQL Mac激活版是一款专门为mac用户推出的数据库管理软件,允许您从一个数据库工具查询,更新,导航和管理所有主要数据库!软件特色RazorSQL 是一个非开源的功能非常强大数据库查询工具、SQL的编辑、数据库管理工具。支持通过 JD…

vsm特征提取java_文本特征提取方案汇总

文本特征提取方案汇总文本分析是机器学习算法的主要应用领域。但是,文本分析的原始数据无法直接丢给算法,这些原始数据是一组符号,因为大多数算法期望的输入是固定长度的数值特征向量而不是不同长度的文本文件。一、文本数据的表示模型​ 文本…

脚本启动显示查询频繁被服务器防御_又被CC攻击弄得心有余悸?莫怕!这里教你如何防御...

转自CSDN,博主:一只IT小小鸟。CC攻击原理HTTP Flood 俗称CC攻击(Challenge Collapsar)是DDOS(分布式拒绝服务)的一种,前身名为Fatboy攻击,也是一种常见的网站攻击方法。是针对 Web 服务在第七层协议发起的攻击。攻击者相较其他三层…

java 怎么获取形参名_获得方法形参名称列表 -- 哦也,搞定!!

JAVA获取类的方法的参数名 – 老话题,新方法!!折腾了一天,终于搞定了.测试了nutz所有的类,均读取正常!! 完美读取任何class的变量名信息! 呵呵,当前,前提是编译时含debug信息.无任何依赖,不需要asm,不要其他任何字节码工具,纯标准JDK API实现. 核心代码,仅一个方法,130行,哦也!…

robotframework安装_python3.9.0 + robotframework + selenium3 实例体验

在win10上安装python3.9.0robotframework中我们做了基本的使用robot framework的环境搭建,这一章主要通过一个简单的实例来体验下robot framework的使用方式、运行、报告和日志(非常漂亮的自动化测试报告噢!)。首先我们打开RIDE,快捷键 ctrln…

vmware 搭建k8s无法ping通子节点_一波四折 —— 记一次K8S集群应用故障排查

一波四折——记一次K8S集群应用故障排查Part1 初露端倪一个周四的下午,客户的报障打破了微信群的平静。“我们部署在自建K8S集群上的应用突然无法正常访问了,现在业务受到了影响!”收到客户的报障,我们立刻响应,向客户…

python有趣的简单代码_简单几步,100行代码用Python画一个蝙蝠侠的logo

转自:菜鸟学Python蝙蝠侠作为DC漫画的核心人物之一,一直都受到广大粉丝的喜爱,而笔者作为DC的铁杆粉丝,自然也是老爷(粉丝对蝙蝠侠的昵称)的支持者。今天,笔者就用Python来画一个蝙蝠侠的logo,大概就是下图…

iis7php怎么301重定向,iis7/8设置网站301重定向的方法

准备条件:a、一台装有win2008以上版本的服务器 b、iis启用并且运行正常 c、在网站程序存放目录中单独创建个目录,目录里面留空即可(为了方便区分,目录名称可以设置为站点名称301,例如fcblog_301)1、打开Internet信息服务…

结构体内元素不确定_查漏补缺!高中三年生物最易忽略、易错的30个知识点整理不容错过...

高中生物的知识体系基本上是由大约数十个核心概念为基础构建起来的,这些概念包括细胞、细胞分裂、光合作用、呼吸作用、基因、染色体、遗传、变异、进化、生化系统等等,今天学姐来帮助你们整理一下高中三年中最容易忽略,也是最容易出错的30个…

基于单片机超声波测距系统的设计_一种基于UWB技术实现的测距防撞系统

叉车被广泛应用于工厂车间、仓库、流通中心和配送中心等,大大提高了对成件托盘货物进行装卸、堆垛和短距离运输作业的运输效率,几乎是所有车间必不可少的运输工具。但目前,简单方便的同时,安全事故(剐蹭、碰撞、碾压、撞车等)却也…

vb.net中递归退到最外层_数组中的逆序对

题目描述在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。 即输出P%1000000007输入描述:题目保证输入的数组中没有的相同的…

java实现条形图,JavaFX条形图

本文概述通常, 条形图可以定义为使用矩形条形表示数据的图。条的长度表示绘制在其中一根轴上的精确数值数据值。矩形条可以在图表上水平或垂直绘制。在下图中, 条形图显示了工程各个分支中的学生人数。 X轴是类别轴, 显示了不同的分支, 而Y轴是数字轴, 显示了特定分支中的学生人…