[ 用大白话讲解复杂的技术 ]

这是我的第 37 篇原创文章作者 l 会点代码的大叔(CodeDaShu)

在学习 Java 反射之前,先让我们看看这几个概念。

01

解释型语言和编译型语言

解释型语言:不需要编译,在运行的时候逐行翻译解释;修改代码时可以直接修改,可以快速部署,不过性能上会比编译型语言稍差;比如 JavaScript、Python ;

编译型语言:需要通过编译器将源代码编译成机器码才能执行;编译之后如果需要修改代码,在执行之前就需要重新编译。比如 C 语言;

Java 严格来说也是编译型语言,但又介于编译型和解释型之间;Java 不直接生成机器码而是生成中间码:编译期间,是将源码交给编译器生成 class 文件(字节码),这个过程中只做了翻译的工作,并没有把代码放入内存运行;当进入运行期,字节码才被 Java 虚拟机加载、解释成机器语言并运行。

02

动态语言和静态语言

动态语言:是指程序在运行时可以改变自身结构,在运行时确定数据类型,一个对象是否能执行某操作,只取决于它有没有对应的方法,而不在乎它是否是某种类型的对象;比如 JavaScript、Python。

静态语言:相对于动态语言来说,在编译时变量的数据类型就已经确定(使用变量之前必须声明数据类型),在编译时就会进行类型是否匹配;比如 C 语言、Java ;

03

反射的概念

Java 反射机制:在运行过程中,对于任意一个类,都能知道其所有的属性和方法;对于任意一个对象,都能调用其属性和方法;这种动态获取类信息和调用对象方法的功能,就是 Java 反射机制。

既然反射里面有一个“反”字,那么我们先看看何为“正”。

在 Java 中,要使用一个类中的某个方法,“正向”都是这样的:

ArrayList list = new ArrayList(); //实例化list.add("reflection");  //执行方法

那么反向(反射)要如何实现?

Class clz = Class.forName("java.util.ArrayList");Method method_add = clz.getMethod("add",Object.class);Constructor constructor = clz.getConstructor();Object object = constructor.newInstance();method_add.invoke(object, "reflection");Method method_get = clz.getMethod("get",int.class);System.out.println(method_get.invoke(object, 0));

两段代码执行的结果是一样的,但是“正向”代码在编译前,就已经明确了要运行的类是什么(ArrayList),而第二段代码,只有在代码运行时,才知道运行的类是 java.util.ArrayList。

04

反射的作用

讲到这里,有些同学可能会有疑问:“反射有什么用?我明明都已经知道了要使用的类是 ArrayList ,我不能直接 new 一个对象然后执行里面的方法么?”

当然可以!不过很多场景中,在代码运行之前并不知道需要使用哪个类,或者说在运行的时候才决定使用哪个类;

比如有这么一个功能:“调用阿里云的人脸识别 API ”;这还不简单,参考对方的 API 文档,很快就能实现。

faceRecognition(Object faceImg){  //调用阿里云的人脸识别 API}

上线一个月后,领导说:“咱公司开始和腾讯云合作了,人脸识别的接口改一下吧”。

faceRecognition(Object faceImg){  //调用腾讯云的人脸识别 API}

修改上线运行了两个月,领导说:“换回来吧”...  ...

当然有聪明的程序员会想到设置一个开关配置,让开关决定走哪段代码逻辑,如果领导哪天想变成亚马逊云的服务,继续写 if-else 就好了:

faceRecognition(Object faceImg){  if("AL".equals(configStr)){    //调用阿里云的人脸识别 API  }else if("TX".equals(configStr)){    //调用腾讯云的人脸识别 API  }else if("AM".equals(configStr)){    //调用亚马逊云的人脸识别 API  }}

不过还有一种更好的方法:

1. 定义一个接口:

interface FaceRecognitionInterface(){  faceRecognition(Object faceImg) ;}

2. 多个实现类:

class ALFaceRecognition implements FaceRecognitionInterface{  //调用阿里云的人脸识别 API 的实现}class TXFaceRecognition implements FaceRecognitionInterface{  //调用腾讯云的人脸识别 API 的实现}

3. 在调用人脸识别功能的代码中:

String configStr = "读取配置,走阿里云还是腾讯云";FaceRecognitionInterface faceRe =  Class.forName(configStr).newInstance();faceRe.faceRecognition(faceImg);

如果上面这个例子,你依然觉得在调用方法中做 if-else 判断,和使用反射实现并没有差太多,但是如果程序员 A 提供接口,程序员 B 提供实现,程序员 C 写客户端呢?

回忆一下 JDBC 的使用,比如创建一个连接:

public Connection getConnection() throws Exception{  Connection conn = null;  //初始化驱动类  Class.forName("com.mysql.jdbc.Driver");  conn = DriverManager.getConnection("jdbc:mysql://url","root", "admin");  return conn;}

其中:

  • 程序员 A 提供接口:Oracle 公司(之前的 Sun)提供 JDBC 标准(接口)。
  • 程序员 B 提供实现:各个数据库厂商提供针对自家数据库的实现。
  • 程序员 C 写客户端:我等码农在 Java 中敲代码访问数据库。

总结一下Java 反射的作用:可以设计出更为通用和灵活的架构,很多框架为了保证其通用性,可以根据配置加载不用的类,这时候要用到反射。除此之外:

  • 动态代理:在不改变目标对象方法的情况下对方法进行增强,比如使用 AOP 拦截某些方法打印日志,这就需要通过反射执行方法中的内容。
  • 注解:利用反射机制,获取注解并执行对应的行为。

05

用反射的用法

上文中我们知道了 Java 运行期的源文件是 class 文件(字节码),所以要使用反射,那么就需要获取到字节码文件对象,在 Java 中,获取字节码文件对象有三种方式:

  • 调用某个类的 class 属性:类名.class
  • 调用对象的 getClass() 方法:对象.getClass()
  • 使用 Class 类中的 forName() 静态方法:Class.forName(类的全路径) ,建议使用这种方法

java.lang.reflect 类库提供了对反射的支持:

  • Field :可以使用 get 和 set 方法读取和修改对象的属性;
  • Method :可以使用 invoke() 方法调用对象中的方法;
  • Constructor :可以用 newInstance() 创建新的对象。

06

反射的优缺点

优点:在运行时动态获取类和对象中的内容,极大地提高系统的灵活性和扩展性;夸张一些说,反射是框架设计的灵魂。

缺点:会有一定的性能损耗,JVM 无法对这些代码进行优化;破坏类的封装性。

总之,可能大家在平时的开发过程中,感觉自己并没有写过反射相关的代码,但是在我们用到的各种开源框架中,反射无处不在。

期待分享

如果您喜欢本文,请点个“在看”或分享到朋友圈,这将是对我最大的鼓励。

java object转泛型_Java 反射:框架设计的灵魂相关推荐

  1. java object转泛型_JAVA快速入门——基本结构、基本数据类型

    Java程序基本结构 一个完整的Java程序基本结构: 下面我们来一点点解剖上面代码的含义: class是关键字,这里定义的class名字就是HelloWorld: 类名必须以英文字母开头,后接字母, ...

  2. Java 反射:框架设计的灵魂

    作者 l 会点代码的大叔(CodeDaShu) 在学习 Java 反射之前,先让我们看看这几个概念. 01 解释型语言和编译型语言 解释型语言:不需要编译,在运行的时候逐行翻译解释:修改代码时可以直接 ...

  3. Java反射:框架设计的灵魂

    Java反射:框架设计的灵魂 框架:办成品软件,可以在框架的基础上进行开发 反射:将类的各个部分封装成对象,这就是反射机制 反射的好处 在程序运行的过程中,操作这些对象 可以降低程序的耦合性,提高程序 ...

  4. java如何用反射把具体方法抽象_如何在Java 中使用泛型或反射机制对DAO进行抽象...

    如何在Java 中使用泛型或反射机制对DAO进行抽象 发布时间:2020-11-26 16:07:42 来源:亿速云 阅读:80 作者:Leah 本篇文章为大家展示了如何在Java 中使用泛型或反射机 ...

  5. Java--反射(框架设计的灵魂)

    Java--反射(框架设计的灵魂) 博客说明 文章所涉及的资料来自互联网整理和个人总结,意在于个人学习和经验汇总,如有什么地方侵权,请联系本人删除,谢谢! 说明 框架:半成品软件.可以在框架的基础上进 ...

  6. java 简单 框架_java简单框架设计

    设计框架包可以作为一个工具给大家用,需要有完全不同设计思路给出来,不同于我们去做一个web服务.网站. 或者一个业务微服务,需要从原来使用视角转换成一个构建者视角. 框架或者工具,更多是框架来管理或者 ...

  7. 【笔记】JAVA中的泛型和反射

    泛型 Java的泛型是如何工作的?什么是类型擦除? ----泛型是通过类型擦除来实现的,编译器在编译时擦除了所有类型相关的信息,所以在运行时不存在任何类型相关的信息.例如List在运行时仅用一个Lis ...

  8. java object转泛型_为什么Java的泛型要用擦除实现

    在 Java 中的 泛型 ,常常被称之为 伪泛型 ,究其原因是因为在实际代码的运行中,将实际类型参数的信息擦除掉了 (Type Erasure) .那是什么原因导致了 Java 做出这种妥协的呢?下面 ...

  9. java模型给泛型_java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一

    对java的泛型特性的了解仅限于表面的浅浅一层,直到在学习设计模式时发现有不了解的用法,才想起详细的记录一下. 1. 概述 泛型在java中有很重要的地位,在面向对象编程及各种设计模式中有非常广泛的应 ...

  10. java 枚举与泛型_Java枚举和泛型

    这件事让我困扰了一段时间.以前我曾问过 questions,可能是一个不好的措辞和一个太抽象的例子.所以不清楚我实际在问什么.我会再尝试.请不要跳过结论.我期待这个问题根本不容易回答! 为什么我不能在 ...

最新文章

  1. Data source rejected establishment of connection, message from server: Too many connections解决办法...
  2. 217. Contains Duplicate - LeetCode
  3. 对于机器学习中,数据增强
  4. 匿名类、包、权限修饰符_DAY10
  5. Django - ORM - 事务, 乐观锁, 悲观锁
  6. master节点重置后添加node报错_超强教程!在树莓派上构建多节点K8S集群!
  7. oracle数据从一个表空间导入到另外一个表空间
  8. 手机app性能测试简介了解
  9. ai python视频教程_2017AI人工智能时代基础实战python机器学习深度学习算法全套视频教程...
  10. 计组实验-CPU设计-指令添加
  11. 神雕侠侣服务器维修时间,《神雕侠侣》2021年3月11日更新维护新服开启公告
  12. AP下发AT+CLAC命令后,手机出现宕机 - MTK物联网在线解答 - 技术论坛
  13. 小鸡腿U T7 NEERC2011
  14. 红孩儿编辑器的模块设计6
  15. 随机数字信号处理实验报告三——Levinson和Burg递推法MATLAB实现
  16. 怎样提高计算机内存,怎样增加电脑虚拟内存
  17. 开水果店前期投入,水果店会有哪些费用
  18. 鹰眼智客地图采集,想采哪里采哪里。
  19. 读Intel Optane DC持久内存模块的基本性能测量有感
  20. 数据集函数sum与单元格函数sum怎么用?

热门文章

  1. Android中的Menu(菜单)的三种类型菜单的学习
  2. t-sql中引号的使用规则 转
  3. C/C++ 输入字符串gets( )、scanf( )、getline( )以及单个字符getchar( )
  4. 【java与移动智能设备】高级用户组件
  5. java 数据透视表 组件_Apache POI如何将工作簿样式应用于所有数据透视表 . 我如何排除床单?...
  6. 给妹子讲python_给妹子讲python-S01E01好用的列表
  7. NPM ---- 配置淘宝镜像CNPM
  8. jmeter如何看tps_jmeter性能测试疑难杂症解决思路
  9. 词组能够进入_四六级翻译100个常考词组~
  10. java 环境win7_java 环境变量 win7