Java面向对象高级【注解和反射】
目录
注解
什么是注解?
自定义注解
元注解
反射
什么是反射
静态语言和动态语言
动态语言
静态语言
对比
Class类
Java内存分析
类加载过程
类加载器
获取运行时类的完整结构
通过Class对象实例化对象
1.调用Class对象的newInstance
2.Constructor类的newInstance
调用对象的方法和属性
调用指定的方法
调用指定的属性
反射操纵注解
注解
什么是注解?
- Annotation是从JDK1.5开始引入的技术
 - Annotation的作用: 
- 不是程序本身,可以对程序做出解释。(和注释差不多)
 - 可以被其他程序读取
 
 - Annotation的格式: 
- 注解是以“@参数名”在代码中存在的,还可以添加一些参数值,比如元注解:@SuppressWarnings(value="unchecked").
 
 - Annotation在哪里使用? 
- 可以附加在package、class、method、field等上面,相当于给他们添加了额外的辅助信息,我们可以同反射机制来实现对这些元数据的访问
 
 
自定义注解
- 使用 @interface 自定义注解时,自动继承了java.lang.annotation.Annotation 接口。
 - 分析 
- @interface 用来声明一个注解,格式:public @interface 注解名 { 方法名()... }
 - 其中每一个方法实际上是一个配置参数
 - 方法名称就是参数名称
 - 返回值类型就是参数类型
 - 可以通过default来声明参数的默认值
 - 如果只有一个参数成员,一般参数名用value
 - 注解必须要有值,我们定义注解时,经常使用空字符串、0作为默认值
 
 
元注解
- 元注解的作用就是负责注解其他注解,Java定义了4个标准的 meta-annotation类型,它们被用来提供对其他annotation类型作说明
 - 这些类型和它们所支持的类在 java.lang.annotation中可以找到(@Target、#Documented、@Inherrited、@Retention) 
- @Target:用于描述注解的使用范围(即注解可以标注在什么地方,类上面或者方法、属性等)
 - @Retention:表示在什么级别保存该注释信息,用于描述注解的声明周期 
- SOURCE:源码阶段-就是写代码的时候]
 - CLASS :表示注解将在编译时被保留,并且会被包含在类文件中,但在运行时不可用。但这意味着,当程序运行时,无法访问该注解。
 - RUNTINUE :程序运行时,可以访问该注解(通过反射)。
 
 - @Documented:说明该注解将被包含在javadoc中。(javadoc是Java语言中自带的一种工具,用于生成API文档。)
 - @Inherited:说明子类可以继承父类的注解
 
 
反射
什么是反射

- Java中的反射是指程序在运行时动态地获取类的信息以及操作类的成员变量、方法和构造方法的能力。
 - 通过反射,可以在运行时检查类的属性和方法,获取类的构造函数并实例化对象,调用类的成员变量和方法,甚至可以在运行时动态地生成新的类,这使得Java程序具有更大的灵活性和动态性。
 - 但是,反射机制也会导致一些性能上的问题,因为反射调用的速度通常比直接调用要慢得多。
 
Java中的反射主要API:
- Class类:用于表示Java类的信息,包括类的名称、父类、接口、构造函数、成员变量和方法等。
 - Constructor类:用于表示Java类的构造函数信息。
 - Method类:用于表示Java类的方法信息。
 - Field类:代表类的成员变量
 
反射机制的核心是在运行时动态地获取类的信息,并通过这些信息来调用类的成员变量和方法,这种能力使得Java程序可以在运行时动态地加载和执行代码,从而实现更加灵活和动态的功能。
静态语言和动态语言
动态语言
- 动态语言是指在运行时进行类型检查的语言。在编写程序时,不需要明确指定变量的数据类型,变量的类型会在运行时根据赋值的内容自动推断。因此,动态语言往往比静态语言更灵活,更容易编写和修改。
 - 常见的动态语言包括Python、JavaScript和Ruby等。
 
静态语言
- 静态语言是指在编译时进行类型检查的语言。在编写程序时,需要明确指定变量的数据类型,并在编译时检查所有变量和表达式的类型是否匹配。如果存在类型不匹配的情况,编译器会报错并终止编译。
 - Java、C++和C#等语言都是静态语言。
 
对比
- 总之,静态语言在编写时需要明确指定类型,编译时会进行类型检查,更加严格和安全,但也更加繁琐;动态语言在编写时不需要指定类型,运行时会自动推断类型,更加灵活和易用,但也更加容易出错。
 
Class类
在Java中,Class类是代表类的实体,它是Java反射机制的核心,用于获取类的信息、创建对象和调用方法等。Class类提供了以下常用的方法:
- getName():获取类的完整名称;
 - newInstance():创建类的实例,等同于使用new关键字构造对象;
 - getConstructors():获取类的所有公共构造器;
 - getMethods():获取类的所有公共方法,包括继承的方法;
 - getDeclaredFields():获取类的所有成员变量,不包括继承的变量;
 - getDeclaredMethods():获取类的所有方法,不包括继承的方法;
 - getSuperclass():获取类的父类;
 - isAssignableFrom(Class c):判断当前类是否可以赋值给参数类c;
 - isInstance(Object obj):判断当前对象是否为指定类的实例;
 - isArray():判断当前类是否为数组类型。
 
Class类的一个重要应用是Java反射机制。通过Class类获取类的信息,可以实现在运行时动态创建对象、调用方法和访问成员变量等功能,增加了程序的灵活性和动态性。同时,反射机制也带来了一定的性能开销,因此应该慎重使用。
无论一个.java文件中有多少个类,最终都只有一个Class对象。
Java内存分析

在Java应用程序中,内存主要分为三个区域:堆、栈和方法区。
- 堆(Heap):用于存储对象实例,堆是Java中最大的内存分配区域。堆内存的大小可以通过-Xmx和-Xms参数来指定,-Xmx表示最大堆内存,-Xms表示初始堆内存。
 - 栈(Stack):用于存储方法调用的信息,每个线程都有自己的栈空间。栈内存的大小是固定的,并且在线程创建时分配。栈中存储着局部变量、方法参数、方法返回值和方法调用的状态等信息。
 - 方法区(Method Area):用于存储类的信息、静态变量、常量等数据。方法区属于堆的一部分,但是它的作用和用途与堆不同。
 
类加载过程

Java的类加载过程分为三个步骤:加载、链接和初始化。其中,加载和链接是在程序运行时进行的,而初始化在类被首次使用时进行。
- 加载:将类的字节码文件加载到内存中,并在内存中创建一个Class对象,用于表示该类的信息。ClassLoader负责查找和加载类的字节码文件,将字节码文件读入内存,并创建Class对象,并将其保存在方法区中。
 - 链接:在链接阶段,虚拟机会对类进行验证、准备和解析。
 - 验证:验证字节码文件的正确性,包括验证文件格式、元数据、字节码和符号引用等是否符合规范。
 - 准备:为类的静态变量分配内存,并设置默认初始值,例如int类型的默认值为0,对象类型的默认值为null。这些内存都在方法区中进行分配。
 - 解析:将符号引用解析为直接引用,例如将方法调用的符号引用解析为实际的方法地址。
 - 初始化:在类被首次使用时进行,虚拟机会执行类的初始化操作,包括执行类的静态代码块和静态变量的初始化。
 - 执行静态代码块:静态代码块是在类被初始化时执行的,它可以用来进行一些静态资源的初始化工作。
 - 静态变量初始化:静态变量的初始化也是在类被初始化时进行的,它可以通过赋初值或静态代码块来进行初始化。
 
类的加载过程是Java虚拟机实现动态性的重要基础,也是Java的重要特性之一。通过自定义ClassLoader,可以实现类的动态加载和替换,这为Java应用程序带来了更大的灵活性和动态性。
类加载器
获取运行时类的完整结构
通过Java反射可以获取类的运行时完整结构,包括类的构造器、方法、字段、注解和泛型等信息,具体步骤如下:
- 获取Class对象:使用 lass.forName() 方法或者类名.class 获取需要反射的类的Class对象。
 - 获取构造器:使用 getConstructors() 方法获取类的所有公共构造器,使用getDeclaredConstructors() 方法获取类的所有构造器(包括私有构造器)。
 - 获取方法:使用 getMethods() 方法获取类的所有公共方法,使用 getDeclaredMethods() 方法获取类的所有方法(包括私有方法)。
 - 获取字段:使用 getFields() 方法获取类的所有公共字段,使用 getDeclaredFields() 方法获取类的所有字段(包括私有字段)。
 - 获取注解:使用 getAnnotations() 方法获取类的所有注解,使用 getDeclaredAnnotations()方法获取类的所有注解(包括私有注解)。
 - 获取泛型:使用 getGenericSuperclass() 方法获取类的带有泛型的父类,使用getGenericInterfaces() 方法获取实现的所有接口(包括带有泛型的接口)。
 
通过反射获取类的运行时完整结构可以实现动态调用、动态代理和动态生成代码等功能,但是反射的性能较低,应该尽量避免在性能要求高的场景中使用。
通过Class对象实例化对象
1.调用Class对象的newInstance
使用newInstance()方法创建类的实例,该方法会调用类的默认构造器来创建对象。
需要注意的是,newInstance()方法只能调用类的默认构造器来创建对象,如果类没有默认构造器或者默认构造器不可访问,则会抛出InstantiationException异常。如果需要调用其他构造器来创建对象,则需要使用Constructor类的newInstance()方法。
2.Constructor类的newInstance
Class<?> c1 = Class.forName("com.reflection.test.User");
//用含参的构造器创建对象Constructor<?> constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);User user1 = (User)constructor.newInstance("张三", 1, 20); 
调用对象的方法和属性
调用指定的方法
通过反射,调用类中的方法,通过Method类完成。
通过Class对象的 getMethod() 或者 getDeclaredMethod() 方法来获得一个Method对象,并设置此方法操作时需要的参数类型。
然后使用 invoke 进行调用
Class<?> c1 = Class.forName("com.refelection.test.User");//获得Class类对象
User user = (User) c1.newInstance();//初始化对象Method setName = c1.getDeclaredMethod("setName", String.class);
setName.invoke(user,"张三");    //调用user的setName方法将user对象的name属性设置为"张三" 
如果方法为private,需要关闭安全检测setAccessible(true)
setName.setAccessible(true); 
调用指定的属性
 //获得属性
Field name = c1.getDeclaredField("name");
name.set("name","张三");//设置属性 
同样如果属性为private,需要关闭安全检测setAccessible(true)
name.setAccessible(true);//关闭权限检测 
反射操纵注解
在Java中,通过反射可以操作注解,具体步骤如下:
- 获取注解信息:使用Class、Method、Field等类的getAnnotation()方法获取注解信息,例如使用getAnnotation()方法获取类、方法或属性上的注解信息,使用getAnnotations()方法获取类、方法或属性上的所有注解信息。
 - 解析注解信息:使用Annotation类的相关方法解析注解信息,例如使用annotationType()方法获取注解的类型,使用value()方法获取注解的属性值等。
 
import java.lang.annotation.*;
import java.lang.reflect.Field;/*** 反射操作注解*/
public class Test12 {public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {//获得Class对象Class<?> c1 = Class.forName("reflection.Student2");//通过反射获得注解Annotation[] annotations = c1.getAnnotations();for (Annotation annotation : annotations) {System.out.println(annotation);//@reflection.TableInfo(value=db_student)}//获得注解的 value 的值TableInfo tableInfo = c1.getAnnotation(TableInfo.class);String value = tableInfo.value();System.out.println(value);//db_student//获得类指定的注解Field f = c1.getDeclaredField("name");FieldInfo fieldInfo = f.getAnnotation(FieldInfo.class);System.out.println(fieldInfo.columnName());//db_nameSystem.out.println(fieldInfo.type());//varcharSystem.out.println(fieldInfo.length());//10}
}@TableInfo("db_student")
class Student2{@FieldInfo(columnName = "db_id",type = "int",length = 10)private int id;@FieldInfo(columnName = "db_age",type = "int",length = 10)private int age;@FieldInfo(columnName = "db_name",type = "varchar",length = 10)private String name;public Student2(){}public Student2(int id, int age, String name) {this.id = id;this.age = age;this.name = name;}public int getId() {return id;}public void setId(int id) {this.id = id;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "Student2{" +"id=" + id +", age=" + age +", name='" + name + '\'' +'}';}
}//类名的注解
@Target(ElementType.TYPE)//作用范围 类
@Retention(RetentionPolicy.RUNTIME)//作用的生命周期 运行时
@interface TableInfo{String value();
}//属性的注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface  FieldInfo{String columnName();String type();int length();
} 
相关文章:
Java面向对象高级【注解和反射】
目录 注解 什么是注解? 自定义注解 元注解 反射 什么是反射 静态语言和动态语言 动态语言 静态语言 对比 Class类 Java内存分析 类加载过程 类加载器 获取运行时类的完整结构 通过Class对象实例化对象 1.调用Class对象的newInstance 2.Constructor…...
Pytorch基础 - 4. torch.expand() 和 torch.repeat()
目录 1. torch.expand(*sizes) 2. torch.repeat(*sizes) 3. 两者内存占用的区别 在PyTorch中有两个函数可以用来扩展某一维度的张量,即 torch.expand() 和 torch.repeat() 1. torch.expand(*sizes) 【含义】将输入张量在大小为1的维度上进行拓展,…...
《LeetCode》——LeetCode刷题日记
本期,将给大家带来的是关于 LeetCode 的关于二叉树的题目讲解。 目录 (一)606. 根据二叉树创建字符串 💥题意分析 💥解题思路 (二)102. 二叉树的层序遍历 💥题意分析 &#…...
mysql数据库审计(1)
1.数据库审计工具介绍及选择 1.1. 数据库审计工具介绍 MySQL 分支的审计功能包含在企业版中,社区版可以使用其他分支提供的工具。目前已知的审计工具,社区版本有 Percona 的 Percona Server Audit Log 、MariaDB 的 MariaDB Audit Plugin 和 McAfee 的…...
Kafka---kafka概述和kafka基础架构
kafka概述和kafka基础架构 文章目录kafka概述和kafka基础架构Kafka定义消息队列传统消息队列应用场景缓存/消峰解耦异步通信消息队列的两种模式点对点模式发布/订阅模式kafka基础架构producerConsumerConsumer Group(CG)BrokerTopicPartitionReplicaLead…...
《JavaEE初阶》多线程基础
《JavaEE初阶》多线程基础 文章目录《JavaEE初阶》多线程基础前言:多线程的概念简单创建线程并运行:简述Thread中run方法与start方法的区别创建线程的几种方法:探讨串行执行与并行执行的执行时间多线程的使用场景:Thread类简单介绍:构造方法:获取线程的常见属性:线程的常用方法…...
技术分享 | OMS 初识
作者:高鹏 DBA,负责项目日常问题排查,广告位长期出租 。 本文来源:原创投稿 *爱可生开源社区出品,原创内容未经授权不得随意使用,转载请联系小编并注明来源。 本文主要贡献者:进行OMS源码分析的…...
【Elastic (ELK) Stack 实战教程】10、ELK 架构升级-引入消息队列 Redis、Kafka
目录 一、ELK 架构面临的问题 1.1 耦合度过高 1.2 性能瓶颈 二、ELK 对接 Redis 实践 2.1 配置 Redis 2.1.1 安装 Redis 2.1.2 配置 Redis 2.1.3 启动 Redis 2.2 配置 Filebeat 2.3 配置 Logstash 2.4 数据消费 2.5 配置 kibana 三、消息队列基本概述 3.1 什么是…...
优先、双端队列-我的基础算法刷题之路(八)
本篇博客旨在整理记录自已对优先队列、双端队列的一些总结,以及刷题的解题思路,同时希望可给小伙伴一些帮助。本人也是算法小白,水平有限,如果文章中有什么错误之处,希望小伙伴们可以在评论区指出来,共勉 &…...
Python3 os.symlink() 方法、Python 质数判断
Python3 os.symlink() 方法 概述 os.symlink() 方法用于创建一个软链接。 语法 symlink()方法语法格式如下: os.symlink(src, dst)参数 src -- 源地址。 dst -- 目标地址。 返回值 该方法没有返回值。 实例 以下实例演示了 symlink() 方法的使用࿱…...
P1972 [SDOI2009] HH的项链
[SDOI2009] HH的项链 题目描述 HH 有一串由各种漂亮的贝壳组成的项链。HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含义。HH 不断地收集新的贝壳,因此,他的项链变得越来…...
力扣解法汇总1026. 节点与其祖先之间的最大差值
目录链接: 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目: https://github.com/September26/java-algorithms 原题链接:力扣 描述: 给定二叉树的根节点 root,找出存在于 不同 节点 A 和 B 之间的最大值…...
010:Mapbox GL移动鼠标mousemove,显示坐标信息
第010个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+mapbox中移动鼠标mousemove,显示坐标信息。 直接复制下面的 vue+mapbox源代码,操作2分钟即可运行实现效果 文章目录 示例效果配置方式示例源代码(共81行)相关API参考:专栏目标示例效果 配置方式 1)查看基础…...
【两阶段鲁棒优化】利用列-约束生成方法求解两阶段鲁棒优化问题(Python代码实现)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...
百度暑期实习 C++ 一面
1.数组 链表 数组是一种线性数据结构,其中相同类型的元素连续存储在一段内存中,并且可以通过索引来访问每个元素。数组的优点是随机访问元素非常快速,但缺点是插入或删除元素可能需要移动其他元素。 链表也是一种线性数据结构,但…...
计算机网络第一章(概述)【湖科大教书匠】
1. 各种网络 网络(Network)由若干**结点(Node)和连接这些结点的链路(Link)**组成多个网络还可以通过路由器互连起来,这样就构成了一个覆盖范围更大的网络,即互联网(互连网)。因此,互联网是"网络的网络(Network of Networks)"**因特…...
【JS】vis.js使用之vis-timeline使用攻略,vis-timeline在vue3中实现时间轴、甘特图
vis.js使用之vis-timeline使用攻略,vis-timeline实现时间轴、甘特图1、vis-timeline简介2、安装插件及依赖3、简单示例4、疑难问题集合1. 中文zh-cn本地化2. 关于自定义class样式无法被渲染3. 关于双向数据绑定vis.js是一个基于浏览器的可视化库,它提供了…...
机器学习——数据处理
机器学习简介 机器学习是人工智能的一个实现途径深度学习是机器学习的一个方法发展而来 机器学习:从数据中自动分析获得模型,并利用模型对未知数据进行预测。 数据集的格式: 特征值目标值 比如上图中房子的各种属性是特征值,然…...
多种文字翻译软件-翻译常用软件
整篇文档翻译软件 整篇文档翻译软件是一种实现全文翻译的自动翻译工具,它能够快速、准确地将整篇文档的内容翻译成目标语言。与单词、句子翻译不同,整篇文档翻译软件不仅需要具备准确的语言识别和翻译技术,还需要考虑上下文语境和文档格式等多…...
Baumer工业相机堡盟工业相机如何通过BGAPI SDK将相机图像数据用二进制的方式保存到本地(C++)
Baumer工业相机堡盟工业相机如何通过BGAPI SDK将相机图像数据用二进制的方式保存到本地(C)Baumer工业相机Baumer工业相机将图像保存为二进制图像的技术背景代码分析第一步:先转换Byte*图像为二进制图像第二步:在回调函数里进行Buf…...
基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
基于服务器使用 apt 安装、配置 Nginx
🧾 一、查看可安装的 Nginx 版本 首先,你可以运行以下命令查看可用版本: apt-cache madison nginx-core输出示例: nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...
家政维修平台实战20:权限设计
目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色…...
Qt Http Server模块功能及架构
Qt Http Server 是 Qt 6.0 中引入的一个新模块,它提供了一个轻量级的 HTTP 服务器实现,主要用于构建基于 HTTP 的应用程序和服务。 功能介绍: 主要功能 HTTP服务器功能: 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...
Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...
自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
Typeerror: cannot read properties of undefined (reading ‘XXX‘)
最近需要在离线机器上运行软件,所以得把软件用docker打包起来,大部分功能都没问题,出了一个奇怪的事情。同样的代码,在本机上用vscode可以运行起来,但是打包之后在docker里出现了问题。使用的是dialog组件,…...
Python基于历史模拟方法实现投资组合风险管理的VaR与ES模型项目实战
说明:这是一个机器学习实战项目(附带数据代码文档),如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融市场日益复杂和波动加剧的背景下,风险管理成为金融机构和个人投资者关注的核心议题之一。VaR&…...
人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...
