Java反射机制

news2024/9/27 9:27:18

目录

反射问题的引出

Java程序在计算机中部署的三个阶段

反射的主要相关类

反射机制的优缺点

调优

反射常用类—Class

特点

常用方法

获取映射Class类对象的四种方式

类加载的三个阶段

加载阶段 Loading

链接阶段 Linking

验证 Verification

准备 Preparation

 解析 Resolution

初始化阶段 Initialization

反射爆破

创建实例

操作属性

操作方法


反射问题的引出

读取一个下述配置文件中指定的信息,创建对象并且调用方法。

利用除反射之外的技术显然是无法在运行阶段动态加载Dog类并创建Dog实例对象的,但是利用Java的反射机制,就可以实现动态加载的效果,并且可以通过修改配置文件在不带便源码的情况下控制程序(例如将配置文件中的method换成别的方法名称,可以不改变源码而调用对象实例的更换的方法):
 

public class Reflect_ {
    public static void main(String[] args) {
        Properties pro = new Properties();
        try {
            pro.load(new FileInputStream("src\\test.properties"));
            String classPath = pro.getProperty("classPath");
            String methodName = pro.getProperty("method");
            //在运行时加载配置文件中Dog类的Class映射类
            Class<?> cls = Class.forName(classPath);
            Object dog = cls.newInstance();   //获得一个Dog的实例对象
            //获得Dog类的映射Class对象中的方法对象
            Method method = cls.getMethod(methodName);
            //调用通过映射创建的Dog实例对象中的hi方法
            method.invoke(dog);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
}

Java程序在计算机中部署的三个阶段

在弄清楚反射机制前,需要先对Java程序的三个部署阶段(编译阶段、加载阶段、运行阶段)进行学习:


反射的主要相关类

与反向相关的类主要有四个:

  • java.lang.Class
    该对象表示某个类加载后在堆中的映射对象,包含了这个类的字段、方法信息
  • java.lang.reflect.Method
    该类包含了某个类在加载后的方法信息
  • java.lang.reflect.Field
    该类包含了某个类在加载后的字段信息
  • java.lang.reflect.Constructor
    该类包含了某个类在加载后的构造方法信息

反射机制的优缺点

优点:可以动态的加载使用对象(也是框架底层的核心),使用灵活
缺点:使用反射基本是解释执行,对程序的运行效率有影响

通过下述验证会发现用于反射动态加载类调用方法的效率大不如静态加载类:

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class Reflect_02 {
    public static void main(String[] args) throws ClassNotFoundException, InvocationTargetException, InstantiationException, IllegalAccessException, NoSuchMethodException {
        tradition();
        reflection();
    }
    //传统方法调用hi
    public static void tradition() {
        Cat cat = new Cat();
        long start = System.currentTimeMillis();    //获取当前系统的相对时间
        System.out.println(start);
        for (int i = 0; i < 900000000; i++) {
            cat.hi();
        }
        long end = System.currentTimeMillis();
        System.out.println(end-start);
    }
    public static void reflection() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        Class<?> cls = Class.forName("com.shuai.Cat");
        Cat cat = (Cat)cls.newInstance();
        Method hi = cls.getMethod("hi");
        //关闭访问检测,优化反射效率
        hi.setAccessible(true); //取消反射调用方法时检查
        long start = System.currentTimeMillis();
        for (int i = 0; i < 900000000; i++) {
            hi.invoke(cat);
        }
        long end = System.currentTimeMillis();
        System.out.println(end-start);
    }
}

//静态加载类的tradition方法的运行时间:4ms
//通过反射动态加载类的reflection方法的运行时间:576ms

调优

针对上述反射机制进行调优可以通过设置不进行反射检查,提高反射程序的运行效率:
在通过某个类的Class映射对象cls获得该类的方法或字段对象后,可以设置方法/字段.setAccessble(true);禁用访问安全检查的开关,提高程序的运行效率。


反射常用类—Class

特点

  • Class类也是一个类,继承自Object(易于其它类混淆
  • Class类对象是在类加载阶段由系统创建的,而不是new出来的;并且Class类对象在内存(堆)中只存在一份,因为类只加载一次
  • 每个类的实例对象都会存储关于自己属于哪个Class映射类的信息,因此也都知道自己属于哪个Class类对象

常用方法

Class<?> cls = Class.forName(classPath);
cls.toString();               //显示自己是哪个类的Class对象
cls.getClass();               //运行类型
cls.getPackage().getName();   //获得关联类的包信息
cls.getName();                //获得映射类的名称
cls.newInstance();            //通过反射创建映射类的实例对象
cls.getField(字段名);          //通过反射获取映射类的字段信息(private字段不能访问到)
field.set(通过反射创建的实例对象,值);    //通过反射给对象字段赋值
field.get(通过反射创建的实例对象);       //获取映射类的属性的值
cls.getMethod(方法名);         //通过反射获得映射类的成员方法信息
method.invoke(通过反射创建的对象);    //通过反射访问映射类的方法
clas.getConstructor(类.class,类.class...);    //获得映射类的构造方法信息

获取映射Class类对象的四种方式

  1. Class.forName(classPath);
    已知一个类的全名称并且该类在当前类路径下,可以使用Class类的静态方法Class.forName(classPath)加载classPath路径下的类的Class对象
    应用场景:多用于配置文件,读取类全路径,加载类
  2. 类名.class
    已知具体的类,可以通过 类.class获取,该方式最安全最可靠,程序的性能最高
    应用场景:多用于参数传递,比如通过反射得到相应的构造器对象,也可用于基本数据类型
  3. 类的实例.getClass( )
    已知某个类的实例,可以调用该实例对象的getClass方法获得该对象所属类的Class对象
    应用场景:实例对象获取类的Class对象
  4. 包装类.TYPE
    基本数据类型对应的包装类,可以通过类名.TYPE获得Class对象

类加载的三个阶段

加载阶段 Loading

JVM将类的字节码从不同的数据源(class文件,jar包,网络等)将数据转化为二进制字节流读入内存,并创建一个java.lang.Class对象

链接阶段 Linking

验证 Verification

确保Class文件的字节流中包含的信息符合当前虚拟机的要求,包括:文件格式验证(是否以魔数0x cafebabe开头)、元数据验证、字节码验证、符号引用验证等。如果项目较大,需要加载的类很多,在此阶段也可以通过 -Xverify:none参数关闭大部分的类的验证机制,缩短虚拟机类加载的时间达到优化效果。

准备 Preparation

JVM在该阶段会对静态变量进行内存分配和默认初始化(初始化为对应数据类型的默认值),静态变量所使用的内存将在方法区中进行分配。

 解析 Resolution

JVM将常量池内的符号引用替换为直接引用。

初始化阶段 Initialization

编译器按照代码语句在源文件中出现的顺序,依次自动收集类中所有的静态变量的赋值动作和静态代码块中的内容并进行合并生成<clinit>方法,并且在此阶段JVM会保证一个类的<clinit>方法在多线程的环境中会被正确加锁,同步,也就是说,如果有多个线程同时初始化一个类,那么只会有一个线程去执行这个类的<clinit>方法,其他线程都会阻塞等待,直到活动线程执行<clinit>方法结束


反射爆破

创建实例

通过反射创建实例有两种方式:

  • 直接调用获取到的反射对象的newInstance方法,该方法通过调用类的public的无参构造创建实例对象。
  • 通过getConstructor() 并且指定参数类型获取该类的公共构造器,通过指定构造器创建对象
  • 通过getDeclaredConstructor() 并且指定参数类型获取该类所有类型的构造器并设置反射爆破创建实例对象:
  • 例如:

操作属性

  • 通过getField()并且指定属性名称可以获取到该类中公共的属性
  • 通过getDeclaredField() 指定属性名称获取该类中所有类型的属性;并且设置反射爆破后,可以访问私有的属性
  • 例如:
     

操作方法

  • 通过getMethod()并且指定形参类型,可以获得类中公共的方法
  • 通过getDeclaredMethod()并指定形参类型可以获得类中所有类型的方法,在设置反射爆破后,可用于调用类中的私有方法
  • 例如:

 

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

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

相关文章

OpenGL示例源码opengl_examples编译

下载好源码并创建编译目录opengl_build 打开CMake-GUI选择源码目录及二进制编译目录:

C#学习记录——【实例】读写ini文件

『知识有两种&#xff0c;一种是你知道的&#xff0c;一种是你知道在哪里能找到的&#xff01;』—— 塞缪尔约翰逊 1、概念 C#读写ini文件之前要了解的概念&#xff1a;INI就是扩展名为"INI"的文件,其实他本身是个文本文件,可以用记事本打开,主要存放的是用户所做…

axios拦截器使用和知识点补充

axios拦截器使用和知识点补充axios拦截器使用axios基地址ajax知识点补充onreadstatechange事件Ajax组成部分了解get请求与post请求区别其他请求方法了解axios拦截器使用 <link rel"stylesheet" href"./lib/bootstrap-v4.6.0.css" /><style>bod…

农业机器人研究进展

文章目录一、农业机器人二、国际农业机器人现状三、我国农业机器人发展情况四、农业机器人展望五、结束语2022年9月17-18日&#xff0c;第十一届中国智能产业高峰论坛成功在厦门举办。大会主论坛上&#xff0c;CAAI副理事长、中国工程院院士、国家农业信息化工程技术研究中心研…

Java实现二叉树

一、树 1、树简介 树是一种非线性的数据结构&#xff0c;具有n个结点其数据存储形式像一棵倒挂的树&#xff0c;树有一个根结点没有前驱结点&#xff0c;树有多个叶子结点没有后继结点&#xff0c;树有多个中间结点既有前驱结点又有后继结点。 树结构中子树之间不能有交集。 n个…

【Node.js实战】一文带你开发博客项目之联调(导入HTML、Nginx反向代理、CORS解决跨域、与前端联调)

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;也会涉及到服务端 &#x1f4c3;个人状态&#xff1a; 在校大学生一枚&#xff0c;已拿多个前端 offer&#xff08;秋招&#xff09; &#x1f680;未…

Tapdata Cloud 场景通关系列: Oracle → MySQL 异构实时同步

【前言】作为中国的 “Fivetran/Airbyte”, Tapdata Cloud 自去年发布云版公测以来&#xff0c;吸引了近万名用户的注册使用。应社区用户上生产系统的要求&#xff0c;Tapdata Cloud 3.0 将正式推出商业版服务&#xff0c;提供对生产系统的 SLA 支撑。Tapdata 目前专注在实时数…

二叉树的遍历(非递归)

二叉树的遍历 遍历二叉树, 是指按一定的规则和顺序访问二叉树的所有结点, 使每一个结点都被访问一次, 而且只被访问一次. 由于二叉树是非线性结构, 因此, 二叉树的遍历实质上是将二叉树的各个结点排列成一个线性序列. DFS: 前序, 中序及后序. BFS: 是指沿着二叉树的宽度优先遍…

Leetcode.1806 还原排列的最少操作步数

题目链接 Leetcode.1806 还原排列的最少操作步数 题目描述 给你一个偶数 ​n​n​n​​​​​ &#xff0c;已知存在一个长度为 nnn 的排列 permpermperm &#xff0c;其中 perm[i]iperm[i] iperm[i]i​&#xff08;下标 从 0 开始 计数&#xff09;。 一步操作中&#xff0…

OLAP和OLTP的区别

OLAP和OLTP的区别 OLAP&#xff1a; (Online transaction processing):在线/联机事务处理。典型的OLTP类操作都比较简单&#xff0c;主要是对数据库中的数据进行增删改查&#xff0c;操作主体一般是产品的用户。 OLTP&#xff1a; (Online analytical processing):指联机分…

Vue新一代状态管理工具—Pinia—都2023年了,快学起来吧!

Pinia 基本介绍 Pinia 是 Vue.js 的轻量级状态管理库 官方网站&#xff1a;https://pinia.vuejs.org/ 中文文档: https://pinia.web3doc.top/introduction.html 为什么学习pinia? pinia和vuex4一样&#xff0c;也是vue 官方 状态管理工具(作者是 Vue 核心团队成员&#xff…

基于JAVA SSM框架的影院管理系统源码,实现包括影院管理,电影管理,影厅管理,排片管理,选座售票,演员管理,影片评论等功能

介绍 下载地址&#xff1a;基于JAVA SSM框架的影院管理系统源码 该项目是一个电影信息管理、发布、展示平台&#xff0c;终端用户可以浏览、购票、评论。项目主要实现包括影院管理&#xff0c;电影管理&#xff0c;影厅管理&#xff0c;排片管理&#xff0c;选座售票&#xff…

连号区间数(第四届蓝桥杯省赛C++B组,第四届蓝桥杯省赛JAVAB组)

题目详细&#xff1a;解题思路&#xff1a;对于这个题目如果一开始没有思路的话我们可以先想一下暴力写法暴力的话就是不断的枚举每个区间然后判断这个区间是否合法这样写下来用了三重循环而对于题目我们只能通过部分样例所以我们就要想办法取缩减它的时间对于遍历每个区间我们…

【SpringBoot】使用AOP+注解实现请求参数的指定自动填充

首先定义一个加在方法上的注解 import java.lang.annotation.*;/*** 开启自动参数填充*/ Retention(RetentionPolicy.RUNTIME) Target({ElementType.METHOD}) Documented Inherited public interface AutoParameterFill {/*** 要填充的字段名,不写的话默认下面类的子类中的字段…

Redis未授权访问漏洞(一)先导篇

前言 Redis默认情况下&#xff0c;会绑定在0.0.0.0:6379&#xff0c;如果没有进行采用相关的策略&#xff0c;比如添加防火墙规则避免其他非信任来源ip访问等&#xff0c;这样将会将Redis服务暴露到公网上。 如果在没有设置密码认证&#xff08;一般为空&#xff09;的情况下…

InceptionNet与ResNet

以下代码图片思路来源&#xff1a; 北京大学Tensor flow笔记 嗯,最近学了一下神经网络&#xff0c;并没有很难&#xff0c;主要是把代码背下来&#xff0c;然后掌握Tensorflow是怎么搭建网络的&#xff0c;Tensorflow是比pytorch好用的&#xff0c;我直接抄的代码里面&#xff…

UDS诊断系列介绍05-27服务

本文框架1. 系列介绍27服务概述2. 27服务请求与应答2.1 27服务请求2.2 27服务肯定应答2.3 27服务否定应答1. 系列介绍 UDS&#xff08;Unified Diagnostic Services&#xff09;协议&#xff0c;即统一的诊断服务&#xff0c;是面向整车所有ECU的一种诊断通信方式&#xff0c;…

java-操作excel

文章目录java操作Excel数据使用场景excel 03 和 07的区别POIeasyExcel解析excel表中的对象POI使用步骤POI 写数据POI 读数据计算公式easyExcel读写数据写数据读数据java操作Excel数据 在 平时 可以使用IO流对Excle进行操作 但是现在使用更加方便的第三方组件来实现 使用场景 1、…

在rhel7系统使用Mariadb

文章目录一 联系和区别二 需求三 部署安装3.1 环境准备3.2 安装软件包3.3 启动服务3.4 设置防火墙策略四 创建用户和库表4.1 登录数据库4.2 创建用户4.3 创建数据库和表五 备份和恢复5.1 备份 com 数据库5.2 模拟误删除操作5.3 恢复表一 联系和区别 Mariadb是由社区开发的一个…

4.4 集成运放的性能指标及低频等效电路

一、集成运放的性能指标 在考察集成运放的性能时&#xff0c;常用下列参数来描述&#xff1a; 1、开环差模增益 AodA_{od}Aod​ 在集成运放无外加反馈时的差模放大倍数称为差模开环增益&#xff0c;记作 AodA_{od}Aod​。AodΔuO/(uP−uN)A_{od}\Delta u_O/(u_P-u_N)Aod​Δ…