Java反射(一)

目录

1.了解反射

2.Class类的三种实例化方法

3.反射机制与对象实例化

4.反射与单例设计模式

5.通过反射获取类结构的信息


1.了解反射

什么是反射,反射有什么作用

1.在Java中,反射是一种机制,允许程序在运行时动态地获取、使用和修改类的信息。通过反射,可以在编译时不知道类的具体信息的情况下,操作和查看类的属性、方法和构造函数等。

2.反射有以下几个主要的作用:

  1. 动态加载类:使用反射可以在运行时动态地加载需要使用的类,而不需要在编译时将类写死在代码中。这样可以实现更加灵活的代码结构和功能。

  2. 获取类的信息:通过反射,可以获取类的名称、父类、接口、方法、字段等详细信息。这对于编写通用框架、调试工具和JavaBean的序列化等场景非常有用。

  3. 创建对象和执行方法:使用反射可以动态地创建对象,即使在编译时无法确定具体的类。同时,还可以在运行时调用任意对象的方法,甚至是私有方法。

  4. 修改私有字段:反射允许程序访问和修改类的私有字段的值。这在某些特定的应用场景中可能是必要的,但需要小心使用,遵循权限和安全性的原则。

需要注意的是,反射是一种强大而复杂的机制,在普通的业务代码中并不常用。滥用反射可能会导致性能下降,代码可读性降低,并增加出错的可能性。因此,在使用反射时,需要谨慎权衡利弊,并确保了解其使用方式和限制。 


2.Class类的三种实例化方法

1.通过实例化对象调用getclass()方法实现

2.通过直接调用类名的.Class

3.通过调用Class.forname("包+类名") throws ClassNoFoundException

实例化案例:

package Example1701;
class Member{private String name;private int age;public void setName(String name) {this.name = name;}public void setAge(int age) {this.age = age;}public String getName() {return name;}public int getAge() {return age;}
}
public class javaDemo {public static void main(String[] args)throws Exception {Member mem = new Member();
//        通过实例化getClass()方法得到类Class<?> claszz1= mem.getClass();System.out.println(claszz1);
//        通过直接获取类名实例化ClassClass<?> claszz2 = Member.class;System.out.println(claszz2);
//        通过调用Class里的forname方法实现实例化Class<?> claszz3 = Class.forName("Example1701.Member");System.out.println(claszz3);}
}

3.反射机制与对象实例化

通过反射机制进行对象实例化就能替代new的关键词的使用

案例

package Example1702;class Member{Member(){System.out.println("自动调用构造函数");}@Overridepublic String toString() {return "实现Member对象的创建";}
}
public class javaDemo {public static void main(String[] args)throws Exception {Class<?> claszz = Class.forName("Example1702.Member");
//        创建实例化对象并用Object类进行接收Object obj = claszz.getDeclaredConstructor().newInstance();System.out.println(obj);
//        对比区别Object obj2 = claszz.newInstance();System.out.println(obj2);}
}

 

问: 可以发现Object接收两个对象输出后都是一样的,那么

Object obj = claszz.getDeclaredConstructor().newInstance();与
Object obj2 = claszz.newInstance();有什么区别吗

答:

在Java 9及之前,我们可以使用Class.newInstance()方法来创建一个类的实例对象。这个方法是通过调用类的默认构造函数来创建对象的。例如,claszz.newInstance()会调用Member类的默认构造函数创建对象。

然而,从Java 9开始,Class.newInstance()方法被标记为过时了,因为它在处理异常和对私有构造函数的访问上存在一些限制。取而代之的是,推荐使用getDeclaredConstructor().newInstance()方法来创建对象。这个方法更为灵活,可以处理带参数的构造函数并且可以处理私有构造函数。

所以,Object obj = claszz.getDeclaredConstructor().newInstance()Object obj2 = claszz.newInstance()的区别在于:

  1. 创建实例对象的方式不同:claszz.getDeclaredConstructor().newInstance()可以处理带参数的构造函数,而claszz.newInstance()只能调用无参构造函数。
  2. 访问权限不同:claszz.getDeclaredConstructor().newInstance()可以处理私有构造函数,而claszz.newInstance()无法处理私有构造函数。
  3. 兼容性不同:claszz.getDeclaredConstructor().newInstance()是在Java 9及之后引入的,而claszz.newInstance()是在Java 8及之前推荐使用的方法。

因此,在现代的Java版本中,建议使用getDeclaredConstructor().newInstance()方法来创建类的实例对象,它更加通用和灵活。


4.反射与单例设计模式

单例设计模式都不陌生,在以前的文章中有提到过,其实现的方法就是通过私有化构造方法实现外部无法实例化对象,并且在类的内部就定义一个唯一的对象,最后通过函数调用的形式将对象输出出去,并且设计模式有两种一种饿汉模式,一种懒汉模式,分别是饿汉:一开始就定义一个唯一对象,懒汉模式:当需要用到的时候才进行创建对象。

那么懒汉模式下就有可能出现问题,什么问题呢?关于多线程问题,懒汉是通过if判断是否对象为空,但是多线程可能出现并发问题,大家都同时执行了if判断语句发现对象为空则大家都一起创建一个对象,造成对象不唯一,不符合单例设计模式

那么如何解决呢?通过反射就能实现多线程下单例设计模式的唯一性

案例代码:

package Example1703;
class Only {
//    实现单例化设计模式private Only(){}private static volatile Only onlyOne = null;
//    创建对象或者获取对象public static  Only getOnly(){if (onlyOne == null){synchronized (Only.class){if (onlyOne == null){onlyOne = new Only();return onlyOne;}}}return onlyOne;}public void print(){System.out.println("Hello");}
}
public class javaDemo {public static void main(String[] args) {
//        创建多线程for (int i = 0;i<3;i++){new Thread(()->{Only.getOnly().print();System.out.println(Thread.currentThread().getName());},i+"的线程").start();}}
}

问1:synchronized (Only.class)的作用,为什么要写Only.class而不是this?

synchronized (Only.class)使用类级别的锁来实现同步。

在Java中,每个类都有一个对应的Class对象,可以通过类名后面加上.class来获取该类的Class对象。而this关键字代表当前实例对象,它是指向当前对象的引用。

在单例模式中,我们希望通过synchronized来保证只有一个线程能够创建实例对象。使用synchronized (Only.class),即使用类级别的锁,意味着多个线程在访问这段代码时会竞争同一个锁,即类级别的锁。

而如果使用synchronized (this),则表示使用实例级别的锁。在这个例子中,我们不希望通过实例级别的锁来控制多个线程对实例的访问,因为还没有创建实例对象,所以也不存在实例对象来获得锁。

因此,为了确保在多线程环境下只创建一个实例对象,需要使用类级别的锁,即synchronized (Only.class),而不是实例级别的锁。

问2:为什么要进行两次判断是否为空?

答: 问2:进行两次判断是否为空的目的是为了提高代码执行的效率和性能。在双重检查锁定机制中,第一次判断onlyOne是否为空是为了避免重复进行同步块的加锁和解锁操作,从而提高了代码的执行效率。如果只有一次判断是否为空,那么每次调用getOnly()方法时都会进入同步块,性能开销会增加。

  同时,第二次判断是为了在多个线程同时通过了第一次判断后,只有第一个获得锁的线程才会进入同步块创建实例。其他线程在获取到锁后,再次判断onlyOne是否为空,如果已经不为空,就直接返回已经创建好的实例,避免了重复创建对象。

如果只有一次判断是否为空,那么即使已经有线程创建了实例,其他线程也会通过第一次判断,进入同步块再次创建实例,这样会造成多个实例的创建,不符合单例模式的要求。

因此,为了确保只有一个线程创建实例,并提高代码的执行效率和性能,需要进行两次判断是否为空。


5.通过反射获取类结构的信息

常用通过反射获取类结构的方法有这几个

方法名作用
getName()获取类的完整名称(包括包路径)
getPackage()获取类所在的包信息
getSuperclass()获取类的父类
getInterfaces()获取类实现的接口列表

案例代码:

package Example1704;interface Interface1 {}
interface Interface2 {}
abstract class Father {}
class Test extends Father implements Interface1, Interface2 {private String name;private int age;Test(String name, int age) {this.age = age;this.name = name;}public void print() {System.out.println("输出信息Test类");}
}public class javaDemo {public static void main(String[] args) {
//        获取指定类的class对象Class<?> clazz = Test.class;//        获取指定类的包的名称Package pag = clazz.getPackage();System.out.println(pag.getName());//        获取类的父类的信息Class<?> superClass = clazz.getSuperclass();System.out.println(superClass.getName());
//        获取类的接口的信息Class<?>[] interfaces = clazz.getInterfaces();for (Class<?> temp : interfaces) {System.out.println(temp.getName());}}
}

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

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

相关文章

ChatGPT学python——制作自己的AI模型(一)初步了解

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 「推荐专栏」&#xff1a; ★java一站式服务 ★ ★前端炫酷代码分享 ★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff0c;vue成神之路★ ★ 解决算法&#xff0c;一个专栏就够了★ ★ 架…

大数据课程E1——Flume的概述

文章作者邮箱&#xff1a;yugongshiyesina.cn 地址&#xff1a;广东惠州 ▲ 本章节目的 ⚪ 了解Ganglia的概念&#xff1b; ⚪ 了解Ganglia的拓扑结构和执行流程&#xff1b; ⚪ 掌握Ganglia的安装操作&#xff1b; 一、简介 1. 概述 1. Flume原本是由Cloude…

【华为HCIP | 高级网络工程师】刷题日记(11)

个人名片&#xff1a; &#x1f43c;作者简介&#xff1a;一名大二在校生&#xff0c;喜欢编程&#x1f38b; &#x1f43b;‍❄️个人主页&#x1f947;&#xff1a;落798. &#x1f43c;个人WeChat&#xff1a;见文末 &#x1f54a;️系列专栏&#xff1a;&#x1f5bc;️ 零…

数据结构和算法入门(时间/空间复杂度介绍--java版)

数据结构和算法入门&#xff08;时间/空间复杂度介绍–java版&#xff09; write in front 作者&#xff1a; 向大佬学习 专栏&#xff1a; 数据结构&#xff08;java版&#xff09; 作者简介&#xff1a;大二学生 希望能学习其同学和大佬的经验&#xff01; 本篇博客简介&…

微信小程序页面传值为对象[Object Object]详解

微信小程序页面传值为对象[Object Object]详解 1、先将传递的对象转化为JSON字符串拼接到url上2、在接受对象页面进行转译3、打印结果 1、先将传递的对象转化为JSON字符串拼接到url上 // info为对象 let stationInfo JSON.stringify(info) uni.navigateTo({url: /pages/statio…

jenkins自定义邮件发送人姓名

jenkins发送邮件的时候发送人姓名默认的&#xff0c;如果要自定义发件人姓名&#xff0c;只需要修改如下信息即可&#xff1a; 系统管理-system-Jenkins Location下的系统管理员邮件地址 格式为&#xff1a;自定义姓名<邮件地址>

VMware虚拟机安装Linux教程(超详细)

一、安装 VMware 官方正版VMware下载&#xff08;16 pro&#xff09;&#xff1a;https://www.aliyundrive.com/drive/file/backup/64c9fa3c132e0d42c60d489c99f3f951ef112ad5 下载Linux系统镜像&#xff08;阿里云盘不限速&#xff09;&#xff1a;https://www.aliyundrive.c…

CCL 2023 电信网络诈骗案件分类评测-第一名方案

1 任务内容 1.1 任务背景 2022年12月1日起&#xff0c;新出台的《反电信网络诈骗犯罪法》正式施行&#xff0c;表明了我国治理当前电信网络诈骗乱象的决心。诈骗案件分类问题是打击电信网路诈骗犯罪过程中的关键一环&#xff0c;根据不同的诈骗方式、手法等将其分类&#xff…

tp-link端口映射设置教程及快解析内网穿透

通常情况下&#xff0c;我们希望互联网的其他用户访问到我们本地局域网内部的一台服务器、监控……等设备或应用&#xff0c;就要在本地路由器或防火墙的出接口/路由器WAN口 上做端口映射&#xff0c;将内部局域网某台计算机的私网IP&#xff0c;如&#xff1a;192.168.1.101 内…

手写一个锁其实也很easy

懵逼的状态&#xff1a; 面试中经常被问到&#xff0c;如何手写一个锁&#xff0c;很多时候一脸懵逼&#xff0c;不知所措&#xff0c;多少年前深有体会&#xff0c;然而回过头来细细分析&#xff0c;只需使用AtomicReference类 即可以轻松搞定。首先咱们先来了解一下Atomi…

数据截断、频谱泄漏与窗函数的选择

目录 数据截断、频谱泄漏与窗函数的选择 什么是频谱泄漏&#xff1f; 解决频谱泄漏问题的方法 主瓣和旁瓣 窗函数介绍 窗函数解决频谱泄漏问题的原理 窗函数的种类、特点和如何使用 1、矩形窗 2、三角窗 3、汉宁窗 4、海明窗 5、布莱克曼窗 6、巴特窗&#xff1a;…

opencv 31-图像平滑处理-方框滤波cv2.boxFilter()

方框滤波&#xff08;Box Filtering&#xff09;是一种简单的图像平滑处理方法&#xff0c;它主要用于去除图像中的噪声和减少细节&#xff0c;同时保持图像的整体亮度分布。 方框滤波的原理很简单&#xff1a;对于图像中的每个像素&#xff0c;将其周围的一个固定大小的邻域内…

MySQL安装 找不到 MSVCP100.dll

安装mysql5.6.51时&#xff0c;出现由于找不到 MSVCP100.dll&#xff0c;无法继续执行代码。重新安装程序可能会解决此问题。 这应该是缺少VS运行库文件导致的&#xff0c;运行库就是支持大部分程序运行的基础&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编…

刷题笔记 day2

力扣 1089 复写零 思路&#xff1a;双指针 第一步&#xff1a;利用指针 cur 去记录最后一位要复写的数 &#xff0c; 利用指针 dest 指向最后一位数所要复写的位置&#xff1b; 实现过程&#xff1a;最开始 cur 指向0&#xff0c;dest 指向 -1 &#xff0c; 当arr[cur] ! …

【UEC++学习】UE网络 - Replication、RPC

1. UE网络架构 &#xff08;1&#xff09;UE的网络架构是SC&#xff08;Server - Client&#xff09;的模式&#xff0c;这种模式的优势&#xff1a;这种模式让所有客户端都在服务器端进行安全验证&#xff0c;这样可以有效的防止客户端上的作弊问题。 &#xff08;2&#xff…

Python(五十六)列表元素的排序操作

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

数据结构初阶--栈和队列

目录 一.栈 1.栈的定义 2.顺序栈的功能实现 2.1.顺序栈的定义 2.2.顺序栈的初始化 2.3.顺序栈的判空 2.4.顺序栈的入栈 2.5.顺序栈的出栈 2.6.顺序栈的取栈顶元素 2.7.顺序栈的求栈的大小 2.8.顺序栈的销毁 2.9.完整程序 Stack.h Stack.c test.c 二.队列 1.队…

使用 Docker Compose 部署 Redis Cluster 集群,轻松搭建高可用分布式缓存

Redis Cluster&#xff08;Redis 集群&#xff09;是 Redis 分布式解决方案的一部分&#xff0c;它旨在提供高可用性、高性能和横向扩展的功能。Redis Cluster 能够将多个 Redis 节点组合成一个分布式集群&#xff0c;实现数据分片和负载均衡&#xff0c;从而确保在大规模应用场…

session反序列化+SoapClientSSRF+CRLF

文章目录 session反序列化SoapClientSSRFCRLF前言bestphps revengecall_user_func()方法的特性SSRFCRLF组合拳session反序列化 解题步骤总结 session反序列化SoapClientSSRFCRLF 前言 从一道题分析通过session反序列化出发SoapClientSSRF利用CRLF解题 bestphp’s revenge 首…

基于方向编码的模板匹配算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 MATLAB2022a 3.部分核心程序 ........................................................................... %选择移动个…