当前位置: 首页 > article >正文

Java自定义注解创建详解

一、什么是自定义注解注解Annotation​ 是一种元数据用来为代码提供额外信息但本身不改变代码逻辑。Java 内置注解如OverrideDeprecatedSuppressWarnings而自定义注解​ 就是开发者自己定义的注解类型用于编译期检查框架配置如 Spring、MyBatis运行时反射处理自动生成代码二、定义一个注解的基本语法public interface 注解名 { 类型 元素名() default 默认值; }注解本质上是一个接口 注解的方法称为元素Element三、元注解必须掌握元注解用来修饰注解的注解。Java 提供了 5 个标准元注解元注解作用Target注解可以用在什么地方Retention注解保留到哪个阶段Documented是否生成到 JavaDocInherited是否允许子类继承Repeatable是否可重复注解1. Target常用Target(ElementType.METHOD)常见取值TYPE 类/接口METHOD 方法FIELDPARAMETERCONSTRUCTORANNOTATION_TYPE2. Retention非常重要Retention(RetentionPolicy.RUNTIME)RetentionPolicy说明SOURCE只在源码中编译后丢弃CLASS编译进 class但运行时不可见RUNTIME运行时可通过反射读取 ✅如果你想在运行期解析注解必须用 RUNTIME四、自定义一个简单注解示例示例 方法级别的日志注解1. 定义注解import java.lang.annotation.*; Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface Log { String value() default ; boolean enable() default true; }说明value()注解的默认属性default默认值支持多个参数2. 使用注解public class UserService { Log(value 查询用户, enable true) public void queryUser() { System.out.println(查询用户逻辑); } Log(删除用户) public void deleteUser() { System.out.println(删除用户逻辑); } }如果只有一个参数且叫value可以省略value3. 运行时解析注解反射import java.lang.reflect.Method; public class AnnotationTest { public static void main(String[] args) throws Exception { Class? clazz UserService.class; for (Method method : clazz.getDeclaredMethods()) { if (method.isAnnotationPresent(Log.class)) { Log log method.getAnnotation(Log.class); System.out.println(方法名 method.getName()); System.out.println(日志描述 log.value()); System.out.println(是否启用 log.enable()); } } } }输出示例方法名queryUser 日志描述查询用户 是否启用true五、自定义注解 AOP 权限校验1项目结构推荐com.example.demo ├── DemoApplication.java ├── annotation │ └── RequiresRole.java ├── aop │ └── RoleAspect.java ├── service │ └── OrderService.java └── controller └── OrderController.java2引入依赖Mavendependencies dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter/artifactId /dependency dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter-aop/artifactId /dependency dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter-web/artifactId /dependency /dependencies3自定义注解核心RequiresRole.javapackage com.example.demo.annotation; import java.lang.annotation.*; Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface RequiresRole { /** * 需要的角色 */ String value(); }关键点RUNTIMEAOP 才能反射拿到用在方法​ 上4AOP 切面重点RoleAspect.javapackage com.example.demo.aop; import com.example.demo.annotation.RequiresRole; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.reflect.MethodSignature; import org.springframework.stereotype.Component; Aspect Component public class RoleAspect { // 当前登录用户模拟 private static final String CURRENT_USER_ROLE USER; Around(annotation(com.example.demo.annotation.RequiresRole)) public Object checkRole(ProceedingJoinPoint pjp) throws Throwable { // 1. 获取方法签名 MethodSignature signature (MethodSignature) pjp.getSignature(); // 2. 获取方法上的注解 RequiresRole requiresRole signature.getMethod().getAnnotation(RequiresRole.class); // 3. 权限校验 if (!CURRENT_USER_ROLE.equals(requiresRole.value())) { throw new RuntimeException(权限不足需要角色 requiresRole.value()); } // 4. 放行 return pjp.proceed(); } }这里做了什么拦截所有带RequiresRole的方法从注解中读取值和当前用户角色比较不通过直接抛异常5业务层使用注解OrderService.javapackage com.example.demo.service; import com.example.demo.annotation.RequiresRole; import org.springframework.stereotype.Service; Service public class OrderService { RequiresRole(ADMIN) public String deleteOrder(Long orderId) { return 订单 orderId 已删除; } RequiresRole(USER) public String queryOrder(Long orderId) { return 查询订单 orderId; } }6Controller 调用OrderController.javapackage com.example.demo.controller; import com.example.demo.service.OrderService; import org.springframework.web.bind.annotation.*; RestController RequestMapping(/order) public class OrderController { private final OrderService orderService; public OrderController(OrderService orderService) { this.orderService orderService; } GetMapping(/delete/{id}) public String delete(PathVariable Long id) { return orderService.deleteOrder(id); } GetMapping(/query/{id}) public String query(PathVariable Long id) { return orderService.queryOrder(id); } }7启动类DemoApplication.javaSpringBootApplication EnableAspectJAutoProxy public class DemoApplication { public static void main(String[] args) { SpringApplication.run(DemoApplication.class, args); } }EnableAspectJAutoProxy必须加8运行结果演示1. 访问有权限GET /order/query/1返回查询订单 12. 访问无权限GET /order/delete/1控制台 / 返回RuntimeException: 权限不足需要角色ADMIN六、注解支持数组类型1 改造注解支持数组RequiresRoles.javaTarget(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface RequiresRoles { /** * 允许访问的角色列表 */ String[] value(); }关键变化String[] value();2 改造 AOP 切面RoleAspect.javaAspect Component public class RoleAspect { // 模拟当前登录用户角色 private static final String CURRENT_USER_ROLE USER; Around(annotation(com.example.demo.annotation.RequiresRoles)) public Object checkRole(ProceedingJoinPoint pjp) throws Throwable { MethodSignature signature (MethodSignature) pjp.getSignature(); RequiresRoles requiresRoles signature.getMethod().getAnnotation(RequiresRoles.class); // 判断是否包含当前用户角色 boolean hasPermission Arrays.asList(requiresRoles.value()) .contains(CURRENT_USER_ROLE); if (!hasPermission) { throw new RuntimeException( 权限不足所需角色 Arrays.toString(requiresRoles.value()) ); } return pjp.proceed(); } }核心逻辑Arrays.asList(roles).contains(currentRole)3 业务方法中使用OrderService.javaService public class OrderService { RequiresRoles({ADMIN, OPS}) public String deleteOrder(Long orderId) { return 订单 orderId 已删除; } RequiresRoles(USER) public String queryOrder(Long orderId) { return 查询订单 orderId; } }4 测试结果当前角色方法结果USERdeleteOrder权限不足ADMINdeleteOrder成功OPSdeleteOrder成功USERqueryOrder成功七、升级版支持 AND / OR 逻辑进阶1注解定义public interface RequiresRoles { String[] value(); Logical logical() default Logical.OR; }public enum Logical { AND, OR }2AOP 判断Logical logical requiresRoles.logical(); ListString requiredRoles Arrays.asList(requiresRoles.value()); if (logical Logical.AND) { // 包含所有角色 } else { // 包含任一角色 }

相关文章:

Java自定义注解创建详解

一、什么是自定义注解注解(Annotation)​ 是一种元数据,用来为代码提供额外信息,但本身不改变代码逻辑。Java 内置注解如:OverrideDeprecatedSuppressWarnings而 自定义注解​ 就是开发者自己定义的注解类型&#xff0…...

弯管LRA计算软件(XYZ转LRA)

专业的“弯管LRA计算软件(XYZ转LRA)”,主要用于将弯管在三维空间中的一系列坐标点(XYZ),转换为管道加工所需的关键制造参数,即LRA(直线段长度、旋转角度、弯曲夹角)。界面…...

Go语言的Web框架:从Gin到Echo

Go语言的Web框架:从Gin到Echo 1. 引言 Web框架是现代Web应用开发的重要工具,它提供了路由、中间件、参数处理等功能,大大简化了Web应用的开发过程。Go语言作为一种高效、简洁的编程语言,拥有丰富的Web框架生态。本文将介绍Go语言…...

别再死记硬背了!用“预测-修正”的直觉理解卡尔曼滤波(附自动驾驶传感器例子)

用“预测-修正”的直觉理解卡尔曼滤波:自动驾驶中的传感器融合艺术 想象一下你在雾天开车,挡风玻璃上沾满雨滴,后视镜模糊不清。此时你需要同时依赖速度表读数、前方车辆尾灯的位置记忆、以及隐约可见的路标来判断自己的位置和速度——这本质…...

C语言void关键字详解:无类型与void指针用法

于C语言里头,“void”属于一种特殊的数据类型,其表明“没有类型”,具体来讲,当我们声明一个函数的返回值类型为“void”之际,我们所指的是该函数不返回任何值,此外地,我们还能够运用“void”指针…...

新手必看:从零到一搞定CTFHub Web入门题(HTTP协议+信息泄露实战)

从零构建CTF Web安全实战能力:HTTP协议与信息泄露攻防指南 当你第一次接触CTF竞赛中那些看似神秘的Web题目时,是否感到无从下手?本文将带你系统掌握Web安全的两大基石——HTTP协议操纵与信息泄露挖掘,通过CTFHub实战平台构建完整的…...

Spring Boot 3.0升级实战:从2.x迁移到3.x的完整避坑指南

Spring Boot 3.0升级实战:从2.x迁移到3.x的完整避坑指南 Spring Boot 3.0的发布为Java开发者带来了诸多令人振奋的新特性,但同时也意味着从2.x版本升级并非简单的版本号变更。本文将深入剖析升级过程中的关键挑战,提供一套经过实战验证的迁移…...

Android OTA包极速提取:payload-dumper-go完整实战指南 [特殊字符]

Android OTA包极速提取:payload-dumper-go完整实战指南 🚀 【免费下载链接】payload-dumper-go an android OTA payload dumper written in Go 项目地址: https://gitcode.com/gh_mirrors/pa/payload-dumper-go payload-dumper-go是一款专为Andro…...

快速原型:用快马ai一键生成openclaw在mac上的自动化安装脚本

最近在Mac上折腾OpenClaw这个开源工具时,发现它的安装过程对新手确实不太友好。作为一个经常需要快速验证工具可行性的开发者,我尝试用InsCode(快马)平台来生成自动化安装脚本,整个过程意外地顺畅。下面分享下我的实践心得: 环境检…...

nli-distilroberta-base保姆级部署教程:开源DistilRoBERTa NLI服务一键启动

nli-distilroberta-base保姆级部署教程:开源DistilRoBERTa NLI服务一键启动 1. 项目介绍 nli-distilroberta-base是一个基于DistilRoBERTa模型的自然语言推理(NLI)Web服务。它能帮你快速判断两个句子之间的关系,特别适合需要分析文本逻辑关系的场景。 …...

DSP28377控制下三相并网系统的双二阶锁相环DSOGI-PLL程序优化及应用

基于DSP28377的三相并网双二阶锁相环DSOGI-PLL程序。系统概述 本文分析的代码实现了一个基于TI DSP28377D处理器的三相并网逆变器控制系统。该系统采用先进的双向功率控制架构,集成了三相锁相环(DSOGI-PLL)、空间矢量脉宽调制(SVPWM)和多种保护机制,适用…...

JAVA无人自助TKV小程序源码实现方案及开源代码片段

无人自助TKV小程序需支持用户自助点歌、支付、设备控制等功能。采用uniapp框架实现跨平台兼容性(微信小程序/H5/Android/iOS),后端使用Spring BootMySQL。关键技术包括:uniapp前端:Vue.js语法uView UI微信支付/支付宝支…...

自我介绍一下

大家好,我是黑名单小羊,是黑客小羊(AI_INT)的小号,希望大家多多观看我的博文,还有黑客小羊的博文,这些都是我最大的动力...

C++笔记 Lambda表达式

Lambda表达式是C11引入的核心特性之一,本质是一种匿名函数,可以捕获作用域内的变量,无需单独定义函数名,就能实现简洁、灵活的代码编写,尤其适合作为回调函数、算法参数(如STL算法)等场景&#…...

网站建设时如何考虑 SEO 因素_如何做好 SEO 竞争对手分析

网站建设时如何考虑 SEO 因素 在现代数字化商业环境中,网站建设不仅仅是一个静态的存在,而是一个动态、竞争激烈的市场。为了在这个竞争中脱颖而出,考虑 SEO 因素是至关重要的。SEO(搜索引擎优化)不仅能够提升网站的可…...

python docker

# Python与Docker:从代码到容器的旅程 在软件开发的世界里,我们常常会遇到这样的场景:代码在开发者的笔记本电脑上运行得完美无缺,但一旦部署到服务器上,就会出现各种莫名其妙的问题。可能是操作系统版本不同&#xff…...

JAVA打车小程序实现原理及开源uniapp代码片段

JAVA打车小程序实现原理打车小程序的核心功能包括用户端、司机端和后台管理系统。用户端实现叫车、订单管理、支付等功能;司机端实现接单、导航、收益管理等功能;后台管理系统负责订单监控、用户管理、数据统计等。用户端功能模块包括地图定位、路线规划…...

python pex

## 聊聊Python的PEX:一个被低估的打包工具 在Python的世界里,打包和分发一直是个让人头疼的问题。传统的pip install虽然方便,但在某些场景下,比如需要快速部署、环境隔离或者离线分发时,就显得有些力不从心了。这时候…...

【12.MyBatis源码剖析与架构实战】9.1 ⼆级缓存的原理

二级缓存(L2 Cache)是计算机体系结构中的关键组件,位于一级缓存(L1)和主内存之间,用于弥合CPU与内存之间的速度差异。下面详细解析其原理,并配合流程图说明数据访问流程。 一、二级缓存的核心原理 1. 存储层次定位 L1缓存:极快(~1ns),极小(32-64KB),与CPU核心紧…...

快速掌握QQ空间历史说说备份:GetQzonehistory完整使用教程

快速掌握QQ空间历史说说备份:GetQzonehistory完整使用教程 【免费下载链接】GetQzonehistory 获取QQ空间发布的历史说说 项目地址: https://gitcode.com/GitHub_Trending/ge/GetQzonehistory 你是否曾经担心QQ空间里的珍贵记忆会随着时间流逝而消失&#xff…...

002.计算机视觉与目标检测发展简史:从传统方法到深度学习

上周调一个老项目,客户要求在不升级硬件的前提下提升夜间车辆检测的准确率。打开代码一看,好家伙,全是手工设计的HOG特征SVM分类器,夜间噪点多的时候误检率直接飙到40%以上。我盯着那些精心调参的边缘梯度直方图代码,突…...

告别英雄联盟繁琐操作:3大核心功能让你轻松掌控游戏节奏

告别英雄联盟繁琐操作:3大核心功能让你轻松掌控游戏节奏 【免费下载链接】League-Toolkit An all-in-one toolkit for LeagueClient. Gathering power 🚀. 项目地址: https://gitcode.com/gh_mirrors/le/League-Toolkit 在快节奏的英雄联盟对局中…...

两相交错并联同步整流双向Buck Boost变换器仿真研究

两相交错并联同步整流双向Buck Boost变换器仿真 所有开关管均可实现ZVs软开关 Buck模式 输入:200-360VDC 额定280VDC 输出:140VDC 10A 开关频率:10kHz Boost模式: 输入:120-160VDC 额定140VDC 输出:280VDC…...

BetterJoy全场景应用指南:从问题诊断到多设备协同的完整解决方案

BetterJoy全场景应用指南:从问题诊断到多设备协同的完整解决方案 【免费下载链接】BetterJoy Allows the Nintendo Switch Pro Controller, Joycons and SNES controller to be used with CEMU, Citra, Dolphin, Yuzu and as generic XInput 项目地址: https://gi…...

Win11Debloat:三分钟搞定Windows 11系统瘦身与隐私保护

Win11Debloat:三分钟搞定Windows 11系统瘦身与隐私保护 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutter and c…...

高温车间降温妙招:蒸发冷省电空调成关键词解决方案

在高温车间工作,闷热的环境不仅影响员工的工作效率,还可能对身体健康造成威胁。因此,为高温车间找到合适的降温方案至关重要。而蒸发冷省电空调,正是解决这一难题的理想选择。蒸发冷省电空调采用先进的制冷技术,通过压…...

Go语言的命令行工具:从flag到cobra

Go语言的命令行工具:从flag到cobra 1. 引言 命令行工具是软件开发中不可或缺的一部分,它们可以帮助我们自动化任务、管理系统、处理数据等。Go语言以其简洁的语法和强大的标准库,成为了开发命令行工具的理想选择。从基础的flag包到高级的co…...

OpCore-Simplify:黑苹果智能配置工具如何化繁为简?

OpCore-Simplify:黑苹果智能配置工具如何化繁为简? 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 为什么黑苹果配置总是让人望…...

如何快速提升Windows性能:Win11Debloat一键优化指南

如何快速提升Windows性能:Win11Debloat一键优化指南 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutter and cust…...

告别90%重复操作:XHS-Downloader如何重构小红书内容采集体验

告别90%重复操作:XHS-Downloader如何重构小红书内容采集体验 【免费下载链接】XHS-Downloader 小红书(XiaoHongShu、RedNote)链接提取/作品采集工具:提取账号发布、收藏、点赞、专辑作品链接;提取搜索结果作品、用户链…...