15. Java反射和注解
Java —— 反射和注解
- 1. 反射
- 2. 注解
1. 反射
| 动态语言:变量的类型和属性可以在运行时动态确定,而不需要在编译时指定 常见动态语言:Python,JavaScript,Ruby,PHP,Perl;常见静态语言(C,C++,C#,Go,Java) Java:Java并不算是严格意义上的动态语言,从反射角度来说属于半动态语言(能通过反射机制实现了部分动态编程的能力) Java反射机制:Java编程语言提供的一种强大的特性,它允许程序在运行时动态地获取类的信息,并通过该信息操作类的成员变量、方法和构造函数 |
Java 的反射机制:在运行状态中,对于任意一个类都能够知道这个类所有的属性和方法;并且对于任意一个对象,都能够调用它的任意一个方法
Java反射常用API
Class 类:反射的核心类,可以获取类的属性,方法等信息
Field 类:Java.lang.reflec 包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值
Method类: Java.lang.reflec 包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法
Constructor 类: Java.lang.reflec 包中的类,表示类的构造方法
获取Class对象的3种方法
调用某个对象的getClass()方法
❶ 类名 对象名 = new 类名();
❷ Class 类对象名 = 对象名.getClass();
使用Class类中的forName()静态方法(安全且性能好)
❸ Class 类对象名 = Class.forName(“类的完全限定名”);
高版本Java创建:var 类对象名 = 类名(需要创建类对象的类).class;
Class<类型(类名)> 类对象名 = 类名(需要创建类对象的类.class;
- 对应类的常用方法




- 对应代码
package reflect;import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;/*** 反射机制*/public class TestDemo {public static void main(String[] args) throws NoSuchMethodException,InvocationTargetException, IllegalAccessException,InstantiationException, NoSuchFieldException {// 创建类对象var classes = Information.class;System.out.println("类名:" + classes.getSimpleName());// 根据类获取当前类的完全限定名String className = classes.getName();System.out.println("当前类的完全限定名:" + className);// 根据获取当前类的包名String packageName = classes.getPackageName();System.out.println("包名:" + packageName);// 根据类获取当前普通方法名List<String> list = new ArrayList<>();Method[] methods = classes.getMethods();for (Method method : methods) {list.add(method.getName());}System.out.println("public方法个数:" + list.size());System.out.println("public方法:" + list);// 获取构造器(无参构造器)Constructor<Information> constructor = classes.getConstructor();// 利用无参构造器创建实例Object obj = constructor.newInstance();// 获取指定的普通方法,带参数Method publicMethod = classes.getMethod("publicMethod", String.class, int.class);// 获取指定私有方法Method privateMethod = classes.getDeclaredMethod("privateMethod", String.class);System.out.println("获取指定私有方法:" + privateMethod);// 强行开启私有方法访问权限privateMethod.setAccessible(true);// 获取所有普通public属性Field[] fields = classes.getFields();System.out.println("普通属性:" + Arrays.toString(fields));// 获取所有属性Field[] declaredFields = classes.getDeclaredFields();System.out.println("所有属性:" + Arrays.toString(declaredFields));// 获取属性名for (Field field:declaredFields) {String name = field.getName();System.out.print(name + ", ");}System.out.println();// 访问单个私有属性Field name = classes.getDeclaredField("name");// 强行打开权限name.setAccessible(true);System.out.println("访问单个私有属性:名称:" + name.getName() + " 类型:" +name.getType() + " 权限修饰符(二进制位标识的修饰符信息):" + name.getModifiers());// 私有方法调用privateMethod.invoke(obj, "零零");// 方法调用publicMethod.invoke(obj, "夏鸥", 22);System.out.println("无参构造器创建的实例:" + obj);// 获取有参构造器Constructor<Information> constructors = classes.getConstructor(String.class, int.class, String.class);// 有参构造器创建实例Object object = constructors.newInstance("王菲", 20, "2020002");// 获取指定的普通方法Method classesMethod = classes.getMethod("publicMethod");// 方法调用classesMethod.invoke(object);System.out.println("有参构造器创建的实例:" + object);}
}class Information{private String name = "赫敏";private int age = 20;public String num = "2020001";public Information() {}public Information(String name, int age, String num) {this.name = name;this.age = age;this.num = num;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getNum() {return num;}public void setNum(String num) {this.num = num;}@Overridepublic String toString() {return "TestDemo{" +"name='" + name + '\'' +", age=" + age +", num='" + num + '\'' +'}';}private void privateMethod() {System.out.println(name + " 私有方法");}private void privateMethod(String name) {this.name = name;System.out.println(name + " 私有方法");}public void publicMethod() {System.out.println(name + " 普通方法");}public void publicMethod(String name, int age) {this.name = name;this.age = age;System.out.println(name + age + " 普通方法");}
}
- 应用实例
package reflect;import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.net.URISyntaxException;
import java.net.URL;/*** 要求:实例化与当前类在统一和包中的所有类*/
public class Instantiations {public static void main(String[] args) throws ClassNotFoundException, URISyntaxException, NoSuchMethodException,InvocationTargetException, InstantiationException, IllegalAccessException {/* 1. 获取当前包中的所有类名 */// 获取包名String packageName = Instantiations.class.getPackageName();// 获取当前绝对路径资源 ".":表示当前路径URL path = Instantiations.class.getResource(".");// 获取路径下的所有文件File file = new File(path.toURI());// 将所需文件过滤存储到数组(所有类)File[] files = file.listFiles(f -> f.getName().endsWith(".class"));// 获取到for (File sub:files) {String name = sub.getName();// 获取类名String className = name.replace(".class", "");System.out.println("实例化类:" + packageName + "." + className);Class<?> classes = Class.forName(packageName + "." + className);// 通过无参构造器创建实例Object obj = classes.getConstructor().newInstance();System.out.println(obj);}}
}
2. 注解
| 注解:元数据 / 注解(注解Annotation是一个接口),一种提供程序中元素信息和数据的途径和方法 作用:在不改变程序主体逻辑的情况下,为程序员提供额外的元数据信息(标记一段代码的功能、作用范围、参数要求等信息) |
4种标准元注解:元注解的作用是负责注解其他注解
❶ @Target 注解修饰范围 / 修饰的目标元素类型
❷ @Retention:注解生命周期 / 保留级别 / 被保留的时间长短
❸ @Documented注解是否会被包含在Java文档中生成
❹ @Inherited注解是否可以被继承
@Target元素类型
ElementType.TYPE:类、接口或枚举类型
ElementType.FIELD:字段或属性
ElementType.METHOD:方法
ElementType.PARAMETER:方法参数
ElementType.CONSTRUCTOR:构造函数
ElementType.LOCAL_VARIABLE:局部变量
ElementType.ANNOTATION_TYPE:注解类型
ElementType.PACKAGE:包
@Retention的三个值
RetentionPolicy.SOURCE:注解仅存在于源代码中,编译时会被丢弃
RetentionPolicy.CLASS:注解存在于源码和编译后的字节码文件中,但在运行时会被丢弃(默认值)
RetentionPolicy.RUNTIME:注解在运行时保留在字节码文件中,可以通过反射机制读取
@Documented和@Inherited
一个注解被@Documented修饰,那么它的信息将会被包含在生成的文档中
一个注解被@Inherited修饰,表示该注解可以被子类继承
- 基本原则:不直接干扰程序代码的运行
Java 注解是一种元数据,它可以提供对程序元素(如类、方法、字段等)的额外信息或配置,并且不会直接影响程序的运行逻辑,只是一种机制,用于存储和传递额外的信息
-
注解分类

-
声明注解及变长参数
package reflect.annotations;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;// 注解类型
@Target(ElementType.METHOD)
// 注解级别
@Retention(RetentionPolicy.RUNTIME)
// 声明注解
public @interface AutoRunMethod {// 注解的参数(当注解下的参数只有一个时,一般用value)int value() default 1;String name() default "张三";
}
- 使用注解
package reflect;import reflect.annotations.AutoRunClass;
import reflect.annotations.AutoRunMethod;import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.URISyntaxException;
import java.util.Arrays;public class Reflect {public static void main(String[] args) {// 加载包下被注解的类被注解的方法try {String packName = Reflect.class.getPackage().getName();File dir = new File(Reflect.class.getResource(".").toURI());File[] subs = dir.listFiles(f->f.getName().endsWith(".class"));for (File sub:subs) {String className = sub.getName().replace(".class","");Class classes = Class.forName(packName + "." + className);if (classes.isAnnotationPresent(AutoRunClass.class)){System.out.println("实例化:" + className);Constructor constructor = classes.getConstructor();Object obj = constructor.newInstance();System.out.println(obj);// 方法名获取Method[] methods = classes.getDeclaredMethods();for (Method method:methods) {// 判断该方法是否被注解if (method.isAnnotationPresent(AutoRunMethod.class)){System.out.println("调用方法:" + method.getName() + "()");// 获取注解参数AutoRunMethod arm = method.getAnnotation(AutoRunMethod.class);// 获取值(定义时将其认为参数,调用时看作方法)int value = arm.value();String name = arm.name();System.out.println("注解参数的值:" + value + name);// 根据注解参数的值调用方法for (int i = 0; i < value; i++) {// 方法调用method.invoke(obj);}}}}}} catch (URISyntaxException | ClassNotFoundException |NoSuchMethodException | InstantiationException |IllegalAccessException | InvocationTargetException e) {throw new RuntimeException(e);}}
}// 注解类
@AutoRunClass
class Student {public String address = "贵阳市花溪区";private String name = "赵涛";private char gender = '男';private int age = 15;public Student() {}public Student(String name, char gender, int age) {this.name = name;this.gender = gender;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public char getGender() {return gender;}public void setGender(char gender) {this.gender = gender;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}// 注解方法@AutoRunMethod(3)public void sayHi() {System.out.println(name + ":你好");}@AutoRunMethod(value = 2, name = "李四")public void sleep() {System.out.println(name + ":睡觉");}@AutoRunMethod(name = "王五")public void watchTV() {System.out.println(name + ":看电视");}public void playGame() {System.out.println(name + ":玩游戏");}public void say(String info) {System.out.println(name + "说:" + info);}public void say(String info, int count) {System.out.println(name + "说了" + count + "次," + info);}public void study() {System.out.println(name + "学习");}@AutoRunMethodpublic void doHomework() {System.out.println(name + ":做作业");}// **********************普通方法(参数变长)只是Java编译器认可(最终为数组)***************************// 变长参数只能在方的最后一个参数,且一个方法只能有一个变长参数public void doSome(String... s) {System.out.println("参数个数:" + s.length);System.out.println("参数:" + Arrays.toString(s));}private void privateMethod() {System.out.println("这是一个私有方法");}@Overridepublic String toString() {return "Student{" +"address='" + address + '\'' +", name='" + name + '\'' +", gender=" + gender +", age=" + age +'}';}
}
- 变长参数只能在方的最后一个参数,且一个方法只能有一个变长参数
- 注解格式:
类型 参数名() [default 默认值]注:不指定默认值时使用注解必须传递对应参数- 注解传参机制:
当注解仅有一个参数,且参数名为value时,直接传入参数(不需要参数名)
当注解仅有一个参数,且参数名不为value时,正常使用注解传参语法:参数名=参数值
多个参数传参使用参数名进行传参,传参顺序可以与注解定义时参数顺序不一致
有多个参数时,即使一个注解的参数名为value,在实际使用时参数名也不可以忽略
相关文章:
15. Java反射和注解
Java —— 反射和注解 1. 反射2. 注解 1. 反射 动态语言:变量的类型和属性可以在运行时动态确定,而不需要在编译时指定 常见动态语言:Python,JavaScript,Ruby,PHP,Perl;常见静态语言…...
pdf处理工具 Enfocus PitStop Pro 2022 中文 for mac
Enfocus PitStop Pro 2022是一款专业的PDF预检和编辑软件,旨在帮助用户提高生产效率、确保印刷品质量并减少错误。以下是该软件的一些特色功能: PDF预检。PitStop Pro可以自动检测和修复常见的PDF文件问题,如缺失字体、图像分辨率低、颜色空…...
微信小程序入门开发教程
🎉🎉欢迎来到我的CSDN主页!🎉🎉 🏅我是Java方文山,一个在CSDN分享笔记的博主。📚📚 🌟推荐给大家我的专栏《微信小程序开发实战》。🎯Ἲ…...
php函数
1. strstr() 返回a在b中的第一个位置 2.substr() 截取字符串 3.PHP字符串函数parse_str(将字符串解析成多个变量)-CSDN博客 4.explode() 字符串分割为数组 5.trim() 1.去除字符串两边的 空白字符 2.去除指定字符 6.extract()函数从数组里…...
3.3 封装性
思维导图: 3.3.1 为什么要封装 ### 3.3.1 为什么要封装 **封装**,在Java的面向对象编程中,是一个核心的思想。它主要是为了保护对象的状态不被外部随意修改,确保数据的完整性和安全性。 #### **核心思想:** - 保护…...
Redis魔法:点燃分布式锁的奇妙实现
分布式锁是一种用于在分布式系统中控制对共享资源的访问的锁。它与传统的单机锁不同,因为它需要在多个节点之间协调以确保互斥访问。 本文将介绍什么是分布式锁,以及使用Redis实现分布式锁的几种方案。 一、前言 了解分布式锁之前,需要先了…...
iOS 项目避坑:多个分类中方法重复实现检测
#前言 在项目中,我们经常会使用分类 -> category。category在实际项目中一般有两个左右:1.给已有class增加方法,扩充起能力、2.将代码打散到多个文件中,避免因为一个类过于复杂而导致代码篇幅过长(应用于viewController中很好用) 但是 category 也有很多弊端~ **首…...
【003】EIS数据分析_#LIB
EIS数据分析 1. EIS测试及数据获取2. EIS数据分析2.1 EIS曲线划分 1. EIS测试及数据获取 点击查看往期介绍 2. EIS数据分析 2.1 EIS曲线划分 一般来说,实轴处的截获表示体电阻(Rb),它反映了电解质,隔膜和电极的电导率。高频区的半圆对应于…...
Sprint framework Day07:注解结合 xml 配置
前言 Spring注解结合XML配置是指在Spring应用中,使用注解和XML配置的方式来进行Bean的定义、依赖注入和其他配置。这种方式可以充分利用Spring框架的注解和XML配置两种不同的配置方式的特点。 在Spring框架中,我们可以使用注解来定义Bean,如…...
LiveGBS流媒体平台GB/T28181功能-国标流媒体服务同时兼容内网收流外网收流多网段设备收流
LiveGBS流媒体平台GB/T28181功能-国标流媒体服务同时兼容内网收流外网收流多网段设备收流 1、背景2、设备接入播放2.1、查看通道2.2、直播播放 3、默认收流地址配置4、其它网络设备收流配置5、搭建GB28181视频直播平台 1、背景 服务器部署的时候,可能有多个网卡多个…...
js题解(四)
文章目录 批量改变对象的属性判断是否包含数字判断是否符合指定格式 批量改变对象的属性 给定一个构造函数 constructor,请完成 alterObjects 方法,将 constructor 的所有实例的 greeting 属性指向给定的 greeting 变量。 function alterObjects(const…...
如何进行大数运算和高精度计算?
大数运算和高精度计算是在计算机编程中常见的需求,尤其是当处理大整数、分数、复数、浮点数等需要更多位数的数据时。在C语言中,由于原生的数据类型有限,您需要使用自定义的数据结构和算法来执行大数运算和高精度计算。在本文中,我…...
身份证读卡器跟OCR有何区别?哪个好?
二代身份证读卡器(以下简称读卡器)和OCR(光学字符识别)是两种常见的身份证信息获取技术,它们在原理、功能和应用方面存在一些区别。下面将详细介绍二者的区别并探讨哪个更好。 1. 原理: - 读卡器ÿ…...
华为云云耀云服务器L实例评测 | 实例评测使用之硬件参数评测:华为云云耀云服务器下的 Linux 网络监控神器 bmon
华为云云耀云服务器L实例评测 | 实例评测使用之硬件参数评测:华为云云耀云服务器下的 Linux 网络监控神器 bmon 介绍华为云云耀云服务器 华为云云耀云服务器 (目前已经全新升级为 华为云云耀云服务器L实例) 华为云云耀云服务器是什…...
C++ 设计模式 —— 组合模式
C 设计模式 —— 组合模式 0. 引用连接 本文主要的思路和代码,来自于对以下连接的学习和实现: 组合模式 1. 引言 1.1 什么是组合模式? 组合模式的定义组合模式的作用 组合模式是一种行为型设计模式,它将对象组合成树形结构以…...
华为云Stack的学习(九)
十、华为云Stack灾备服务介绍 1.云硬盘备份VBS 云硬盘备份服务(VBS,Volume Backup Service)可为云硬盘(EVS,Elastic Volume Service)创建备份,利用备份数据恢复云硬盘,最大限度保障…...
Flink中jobmanager、taskmanager、slot、task、subtask、Parallelism的概念
场景 一个工厂有三个车间每个车间两条生产线 生产流程如下 原料->加工->过滤->分类->美化->包装->下线 JobManager:工厂 在上述场景中,工厂就是jobManager,负责协调、调度和监控整个生产过程 TaskManager:车间…...
OpenHarmony docker环境搭建
OpenHarmony docker环境搭建 要求一台安装ubuntu的虚拟机,vscode软件 安装docker 在 Ubuntu 上安装 Docker 非常直接。我们将会启用 Docker 软件源,导入 GPG key,并且安装软件包。 首先,更新软件包索引,并且安装必要的依赖软件…...
【计算机网络】网络编程接口 Socket API 解读(11)
Socket 是网络协议栈暴露给编程人员的 API,相比复杂的计算机网络协议,API 对关键操作和配置数据进行了抽象,简化了程序编程。 本文讲述的 socket 内容源自 Linux man。本文主要对各 API 进行详细介绍,从而更好的理解 socket 编程。…...
Qt工具开发,该不该跳槽?
Qt工具开发,该不该跳槽? 就这样吧,我怕你跳不动。 嵌入式UI,目前趋势是向着LVGL发展。QT已经在淘汰期了。很多项目还在用,但技术上已经落后。QT短期内不会全面淘汰,但退位让贤的大趋势已经很清楚了。 最近很多小伙伴…...
(十)学生端搭建
本次旨在将之前的已完成的部分功能进行拼装到学生端,同时完善学生端的构建。本次工作主要包括: 1.学生端整体界面布局 2.模拟考场与部分个人画像流程的串联 3.整体学生端逻辑 一、学生端 在主界面可以选择自己的用户角色 选择学生则进入学生登录界面…...
VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
微软PowerBI考试 PL300-选择 Power BI 模型框架【附练习数据】
微软PowerBI考试 PL300-选择 Power BI 模型框架 20 多年来,Microsoft 持续对企业商业智能 (BI) 进行大量投资。 Azure Analysis Services (AAS) 和 SQL Server Analysis Services (SSAS) 基于无数企业使用的成熟的 BI 数据建模技术。 同样的技术也是 Power BI 数据…...
页面渲染流程与性能优化
页面渲染流程与性能优化详解(完整版) 一、现代浏览器渲染流程(详细说明) 1. 构建DOM树 浏览器接收到HTML文档后,会逐步解析并构建DOM(Document Object Model)树。具体过程如下: (…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
智能分布式爬虫的数据处理流水线优化:基于深度强化学习的数据质量控制
在数字化浪潮席卷全球的今天,数据已成为企业和研究机构的核心资产。智能分布式爬虫作为高效的数据采集工具,在大规模数据获取中发挥着关键作用。然而,传统的数据处理流水线在面对复杂多变的网络环境和海量异构数据时,常出现数据质…...
Fabric V2.5 通用溯源系统——增加图片上传与下载功能
fabric-trace项目在发布一年后,部署量已突破1000次,为支持更多场景,现新增支持图片信息上链,本文对图片上传、下载功能代码进行梳理,包含智能合约、后端、前端部分。 一、智能合约修改 为了增加图片信息上链溯源,需要对底层数据结构进行修改,在此对智能合约中的农产品数…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
Kubernetes 网络模型深度解析:Pod IP 与 Service 的负载均衡机制,Service到底是什么?
Pod IP 的本质与特性 Pod IP 的定位 纯端点地址:Pod IP 是分配给 Pod 网络命名空间的真实 IP 地址(如 10.244.1.2)无特殊名称:在 Kubernetes 中,它通常被称为 “Pod IP” 或 “容器 IP”生命周期:与 Pod …...
