java如何使用注解实现自定义拦截器 java注解拦截器实现的基础入门教程​

定义注解:创建一个运行时保留、作用于方法的自定义注解(如@loggable),用于标记需要拦截的方法;2. 编写切面:通过@aspect和@component定义切面类,使用@pointcut指定匹配规则,结合@around等通知类型实现环绕拦截逻辑,利用反射获取方法信息并执行前置、后置及异常处理操作;3. 配置与使用:在spring boot中引入aop依赖后自动启用代理,无需显式配置@enableaspectjautoproxy,直接在目标方法上添加自定义注解即可生效;4. 优势分析:注解方式具有声明式、非侵入性特点,避免了传统xml配置或接口实现带来的代码污染,提升业务逻辑纯净度与可维护性;5. 常见问题:包括内部方法调用不触发拦截(因代理失效)、多切面执行顺序需用@order控制、性能开销在高并发场景下需评估、以及过度使用导致代码隐式性增强增加调试难度;6. 扩展应用:除日志外,还可用于权限校验(如@requirespermission)、参数验证、性能监控、幂等控制、数据脱敏、接口限流和缓存管理等横切关注点,显著提升系统模块化程度与开发效率。

java如何使用注解实现自定义拦截器 java注解拦截器实现的基础入门教程​

在Java中,利用注解实现自定义拦截器,本质上是运用了面向切面编程(AOP)的思想,通过定义一套声明式的规则,让程序在特定执行点(比如方法调用前后)自动插入预设的逻辑。这提供了一种极其优雅且非侵入性的方式来处理横切关注点,比如日志记录、权限校验、性能监控或事务管理,使得核心业务逻辑保持高度纯净,极大提升了代码的可读性、可维护性和模块化程度。

解决方案

要构建一个基于注解的自定义拦截器,我们通常会经历以下几个核心步骤:定义注解、编写切面(即拦截器逻辑)、以及配置(在Spring Boot等框架下通常是自动配置)。

首先,我们需要一个自定义注解来标记那些我们希望被拦截的方法或类。这个注解需要指定其作用范围(方法、类等)和生命周期。

立即学习“Java免费学习笔记(深入)”;

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 Loggable {    String value() default ""; // 可以带一个参数,比如日志描述}

接着,就是编写实际的拦截逻辑了。这通常通过一个切面(Aspect)来完成,我们会在其中定义“切点”(Pointcut)来匹配带有我们自定义注解的方法,并定义“通知”(Advice)来指定在切点执行前、后或环绕执行的逻辑。这里以Spring AOP为例:

import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Pointcut;import org.aspectj.lang.reflect.MethodSignature;import org.springframework.stereotype.Component;import java.lang.reflect.Method;import java.util.Arrays;/** * 日志拦截切面 */@Aspect // 声明这是一个切面@Component // 注册为Spring组件public class LoggableAspect {    // 定义切点,匹配所有带有 @Loggable 注解的方法    @Pointcut("@annotation(com.example.demo.annotation.Loggable)")    public void loggablePointcut() {    }    /**     * 环绕通知:在方法执行前、后都执行逻辑     * @param joinPoint 连接点,包含了被拦截方法的信息     * @return 方法执行结果     * @throws Throwable 异常     */    @Around("loggablePointcut()")    public Object logMethodExecution(ProceedingJoinPoint joinPoint) throws Throwable {        long startTime = System.currentTimeMillis();        MethodSignature signature = (MethodSignature) joinPoint.getSignature();        Method method = signature.getMethod();        String methodName = method.getName();        String className = method.getDeclaringClass().getSimpleName();        Object[] args = joinPoint.getArgs();        // 获取注解上的值        Loggable loggable = method.getAnnotation(Loggable.class);        String logDescription = loggable != null ? loggable.value() : "无描述";        System.out.println(">>> 进入方法: " + className + "." + methodName + "(), 描述: " + logDescription + ", 参数: " + Arrays.toString(args));        Object result = null;        try {            // 执行目标方法            result = joinPoint.proceed();            System.out.println("<<< 离开方法: " + className + "." + methodName + "(), 返回值: " + result);        } catch (Throwable e) {            System.err.println("!!! 方法异常: " + className + "." + methodName + "(), 异常信息: " + e.getMessage());            throw e; // 重新抛出异常,让上层处理        } finally {            long endTime = System.currentTimeMillis();            System.out.println("--- 方法执行耗时: " + (endTime - startTime) + "ms");        }        return result;    }}

最后,在你的Spring Boot应用中,你只需要确保你的主应用类上带有

@EnableAspectJAutoProxy

注解(在Spring Boot中,如果引入了

spring-boot-starter-aop

依赖,这个通常是自动配置的),然后就可以直接在需要拦截的方法上使用

@Loggable

注解了。

import org.springframework.stereotype.Service;import com.example.demo.annotation.Loggable; // 假设你的注解包路径@Servicepublic class MyBusinessService {    @Loggable("处理用户订单")    public String processOrder(String orderId, int quantity) {        System.out.println("实际业务逻辑:正在处理订单 " + orderId + ",数量 " + quantity);        if (quantity <= 0) {            throw new IllegalArgumentException("数量必须大于0");        }        return "订单 " + orderId + " 处理成功";    }    @Loggable("查询商品信息")    public String getProductInfo(String productId) {        System.out.println("实际业务逻辑:查询商品 " + productId);        return "商品[" + productId + "] 详情";    }    public void doSomethingElse() {        System.out.println("这个方法没有注解,不会被拦截。");    }}

运行程序,调用

processOrder

getProductInfo

方法时,你就能看到切面中定义的日志输出了。

为什么选择注解来实现Java拦截器,它比传统方式有哪些优势?

选择注解来实现Java拦截器,在我看来,最显著的优势在于其无与伦比的“声明式”与“非侵入性”。想想看,过去我们可能需要通过XML配置一大堆Bean、代理,或者让业务类实现特定的接口,然后通过工厂模式来获取代理对象。这不仅让配置变得臃肿,更重要的是,它污染了业务代码。业务代码为了被拦截,不得不去感知拦截器的存在,这显然违背了单一职责原则。

注解则彻底改变了这一点。它就像给方法或类贴上了一个标签,告诉AOP框架:“嘿,这里需要特殊处理!”。业务代码本身不需要知道具体如何处理,也不需要引入额外的接口或基类。这种解耦使得业务逻辑保持了极高的纯净度,提高了代码的可读性和维护性。比如,一眼看到

@Loggable

,我就知道这个方法有日志切面在工作,而不需要去翻阅复杂的XML文件或者查看继承关系。

此外,注解方式也极大地简化了开发流程。在Spring Boot这类框架下,很多AOP配置都是自动化的,开发者只需要定义好注解和切面,几乎无需额外配置就能让拦截器生效。这比手动管理代理、配置拦截链条要高效太多了。它把原本复杂且重复的横切逻辑,转化成了一种直观、简洁的元数据声明。

自定义注解拦截器在实际开发中可能遇到哪些常见问题与挑战?

虽然注解拦截器非常强大,但在实际开发中,它也不是万能药,总会碰到一些让人挠头的问题。

一个很常见的坑是“内部方法调用不被拦截”。比如,在一个Service类中,

methodA()

调用了同一个Service实例的

methodB()

,如果

methodB()

上有注解,但

methodA()

没有,那么

methodB()

的拦截器可能不会生效。这是因为Spring AOP默认是基于动态代理实现的(JDK动态代理或CGLIB),它只对外部调用有效。当

methodA()

调用

methodB()

时,它实际上是直接调用了原始对象的方法,而不是代理对象的方法。解决这个问题,通常需要通过

AopContext.currentProxy()

获取当前代理对象再调用,或者考虑引入AspectJ的编译时/加载时织入。

另一个挑战是“拦截器执行顺序”。当多个切面都匹配同一个方法时,它们的执行顺序可能会变得复杂。Spring AOP提供

@Order

注解来指定切面的优先级,数字越小优先级越高。但如果设计不当,不同的切面之间可能会产生意想不到的交互,甚至导致死循环或逻辑错误。这要求开发者在设计多个切面时,必须清晰地规划好它们的职责和执行顺序。

还有就是“性能开销”。虽然现代JVM和AOP框架对反射和代理的优化已经非常出色,但在极端高并发或对性能要求极高的场景下,过多的拦截器层层嵌套,依然可能带来额外的性能损耗。这通常不是一个大问题,但值得在性能敏感的模块中进行考量和测试。

最后,过度依赖AOP可能导致“代码的隐式性”增加,也就是所谓的“魔法”代码。虽然注解让代码看起来很干净,但如果滥用,开发者可能会忘记某个注解背后隐藏着复杂的逻辑,导致调试困难。当出现问题时,追踪会变得更长,更难定位到真正的业务逻辑错误。这要求我们在使用AOP时保持克制,只将真正属于横切关注点的逻辑放入其中。

除了基础的日志记录,自定义注解拦截器还能在哪些业务场景中发挥作用?

自定义注解拦截器的应用场景远不止日志记录那么简单,它几乎可以用于所有需要“横切”到业务逻辑的非功能性需求。

一个非常典型的应用是权限控制。我们可以定义一个

@RequiresPermission("user:create")

这样的注解,然后编写一个切面,在方法执行前检查当前用户是否拥有所需的权限。如果没有,就抛出未授权异常。这比在每个业务方法内部写

if (hasPermission())

要优雅得多,且易于统一管理。

参数校验也是一个很好的例子。设想你有一个方法接收一个用户ID,你希望确保这个ID不是空的或者符合某种格式。你可以定义一个

@ValidateUserId

注解,并在拦截器中实现校验逻辑。这样,业务方法就不需要再重复这些校验代码了。

性能监控同样非常适合。定义一个

@MonitorPerformance

注解,拦截器可以在方法执行前后记录时间戳,计算方法的执行耗时,甚至将数据发送到监控系统,帮助我们发现性能瓶颈。

在分布式系统中,幂等性处理也常常借助于注解拦截器。例如,对于一个提交订单的接口,可以定义

@Idempotent

注解,拦截器在方法执行前检查请求的唯一标识符(如请求ID),确保同一个请求不会被重复处理,有效防止重复提交。

此外,还有数据脱敏/加密(如

@SensitiveData

自动对返回结果中的敏感字段进行脱敏)、API限流(如

@RateLimit(permits = 10)

控制方法每秒调用次数)、以及缓存管理(虽然Spring自带了

@Cacheable

等,但其底层原理也是AOP,我们可以根据自己的需求定制更复杂的缓存策略)等。这些场景都受益于注解拦截器提供的声明式、非侵入性特性,极大地提升了开发效率和系统健壮性。

以上就是java如何使用注解实现自定义拦截器 java注解拦截器实现的基础入门教程​的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/118988.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月30日 03:01:33
下一篇 2025年11月30日 03:22:54

相关推荐

  • Linux journalctl与systemctl status结合分析

    先看 systemctl status 确认服务状态,再用 journalctl 查看详细日志。例如 nginx 启动失败时,systemctl status 显示 Active: failed,journalctl -u nginx 发现端口 80 被占用,结合两者可快速定位问题根源。 在 Lin…

    2025年12月6日 运维
    100
  • Linux如何防止缓冲区溢出_Linux防止缓冲区溢出的安全措施

    缓冲区溢出可通过栈保护、ASLR、NX bit、安全编译选项和良好编码实践来防范。1. 使用-fstack-protector-strong插入canary检测栈破坏;2. 启用ASLR(kernel.randomize_va_space=2)随机化内存布局;3. 利用NX bit标记不可执行内存页…

    2025年12月6日 运维
    000
  • “史上最强Ace”来袭!一加 Ace 6携7800mAh电池和165Hz屏幕打造满配旗舰

    10月23日,一加官方宣布将于10月27日正式推出全新机型——一加 ace 6。一加中国区总裁李杰在预热中称其为“史上最强ace”,并强调这是一款真正意义上的满血旗舰,涵盖了性能、续航、屏幕、防护等级和机身质感等全方位顶级配置,“能给的全都给到位”。 图片来源微博@李杰Louis 据官方信息显示,一…

    2025年12月6日 行业动态
    000
  • Vue.js应用中配置环境变量:灵活管理后端通信地址

    在%ignore_a_1%应用中,灵活配置后端api地址等参数是开发与部署的关键。本文将详细介绍两种主要的环境变量配置方法:推荐使用的`.env`文件,以及通过`cross-env`库在命令行中设置环境变量。通过这些方法,开发者可以轻松实现开发、测试、生产等不同环境下配置的动态切换,提高应用的可维护…

    2025年12月6日 web前端
    000
  • VSCode选择范围提供者实现

    Selection Range Provider是VSCode中用于实现层级化代码选择的API,通过注册provideSelectionRanges方法,按光标位置从内到外逐层扩展选择范围,如从变量名扩展至函数体;需结合AST解析构建准确的SelectionRange链式结构以提升选择智能性。 在 …

    2025年12月6日 开发工具
    000
  • JavaScript动态生成日历式水平日期布局的优化实践

    本教程将指导如何使用javascript高效、正确地动态生成html表格中的日历式水平日期布局。重点解决直接操作`innerhtml`时遇到的标签闭合问题,通过数组构建html字符串来避免浏览器解析错误,并利用事件委托机制优化动态生成元素的事件处理,确保生成结构清晰、功能完善的日期展示。 在前端开发…

    2025年12月6日 web前端
    000
  • JavaScript响应式编程与Observable

    Observable是响应式编程中处理异步数据流的核心概念,它允许随时间推移发出多个值,支持订阅、操作符链式调用及统一错误处理,广泛应用于事件监听、状态管理和复杂异步逻辑,提升代码可维护性与可读性。 响应式编程是一种面向数据流和变化传播的编程范式。在前端开发中,尤其面对复杂的用户交互和异步操作时,J…

    2025年12月6日 web前端
    000
  • JavaScript生成器与迭代器协议实现

    生成器和迭代器基于统一协议实现惰性求值与数据遍历,通过next()方法返回{value, done}对象,生成器函数简化了迭代器创建过程,提升处理大数据序列的效率与代码可读性。 JavaScript中的生成器(Generator)和迭代器(Iterator)是处理数据序列的重要机制,尤其在处理惰性求…

    2025年12月6日 web前端
    000
  • 如何在mysql中分析索引未命中问题

    答案是通过EXPLAIN分析执行计划,检查索引使用情况,优化WHERE条件写法,避免索引失效,结合慢查询日志定位问题SQL,并根据查询模式合理设计索引。 当 MySQL 查询性能下降,很可能是索引未命中导致的。要分析这类问题,核心是理解查询执行计划、检查索引设计是否合理,并结合实际数据访问模式进行优…

    2025年12月6日 数据库
    000
  • VSCode入门:基础配置与插件推荐

    刚用VSCode,别急着装一堆东西。先把基础设好,再按需求加插件,效率高还不卡。核心就三步:界面顺手、主题舒服、功能够用。 设置中文和常用界面 打开软件,左边活动栏有五个图标,点最下面那个“扩展”。搜索“Chinese”,装上官方出的“Chinese (Simplified) Language Pa…

    2025年12月6日 开发工具
    000
  • VSCode性能分析与瓶颈诊断技术

    首先通过资源监控定位异常进程,再利用开发者工具分析性能瓶颈,结合禁用扩展、优化语言服务器配置及项目设置,可有效解决VSCode卡顿问题。 VSCode作为主流的代码编辑器,虽然轻量高效,但在处理大型项目或配置复杂扩展时可能出现卡顿、响应延迟等问题。要解决这些性能问题,需要系统性地进行性能分析与瓶颈诊…

    2025年12月6日 开发工具
    000
  • VSCode的悬浮提示信息可以自定义吗?

    可以通过JSDoc、docstring和扩展插件自定义VSCode悬浮提示内容,如1. 添加JSDoc或Python docstring增强信息;2. 调整hover延迟与粘性等显示行为;3. 使用支持自定义提示的扩展或开发hover provider实现深度定制,但无法直接修改HTML结构或手动编…

    2025年12月6日 开发工具
    000
  • 优化PDF中下载链接的URL显示:利用HTML title 属性

    在pdf文档中,当包含下载链接时,完整的url路径通常会在鼠标悬停时或直接显示在链接文本中,这可能不符合预期。本文将探讨为何传统方法如`.htaccess`重写或javascript不适用于pdf环境,并提出一种利用html “ 标签的 `title` 属性来定制链接悬停显示文本的解决方…

    2025年12月6日 后端开发
    000
  • Phaser 3 游戏画布响应式适配:保持高度控制宽度

    本文旨在提供一种在 Phaser 3 游戏中实现画布响应式适配的方案,核心思路是利用 `Phaser.Scale.HEIGHT_CONTROLS_WIDTH` 缩放模式,使画布高度适应父容器,宽度随之调整,并始终居中显示。这种方法适用于需要保持游戏核心内容在屏幕中央,允许左右裁剪的场景。 在 Pha…

    2025年12月6日 web前端
    000
  • 在 Java 中使用 Argparse4j 接收 Duration 类型参数

    本文介绍了如何使用 `net.sourceforge.argparse4j` 库在 Java 命令行程序中接收 `java.time.Duration` 类型的参数。由于 `Duration` 不是原始数据类型,需要通过自定义类型转换器或工厂方法来处理。文章提供了两种实现方案,分别基于 `value…

    2025年12月6日 java
    000
  • VSCode插件:GitLens使用详解

    GitLens是VSCode中强大的Git增强插件,提供行级代码追踪、提交历史浏览、版本对比、跨文件导航及与GitHub等平台集成;通过启用Current Line Blame和In-Line Blame,可实时查看每行代码的作者与修改时间;支持按分支、作者过滤提交记录,比较差异,并利用Go Bac…

    2025年12月6日 开发工具
    000
  • 使用 String 和 Enum 的 Switch Case 详解

    本文详细讲解了如何在 Java 中结合 String 和 Enum 类型进行 switch case 操作。重点介绍了如何将字符串转换为 Enum 类型,以及如何在 switch 语句中使用 Enum。同时,探讨了分离关注点的原则,并提供了一个完整的示例,展示了如何将字符串到 Enum 的映射与实际…

    2025年12月6日 java
    000
  • mysql如何备份存储过程和函数

    最直接且推荐的方式是使用mysqldump工具并添加–routines参数,可完整导出存储过程和函数;若需跨版本迁移,应结合–triggers、处理DEFINER用户、验证SQL_MODE,并在测试环境充分验证恢复与兼容性。 MySQL备份存储过程和函数,最直接且推荐的方式是…

    2025年12月6日 数据库
    000
  • VSCode调试:快速定位与修复问题

    掌握VSCode调试技巧可提升开发效率。首先设置断点并配置launch.json文件,通过“运行和调试”面板启动调试;程序暂停时利用变量窗格查看数据状态,结合调用栈追溯函数执行路径;使用调试控制台动态执行代码、验证逻辑;针对高频调用场景,可设置条件断点(如i===100)或日志断点输出信息而不中断执…

    2025年12月6日 开发工具
    000
  • 洋葱浏览器下载文件安全吗_使用洋葱浏览器安全下载文件的注意事项

    首先验证.onion链接真实性,通过可信渠道获取并核对PGP签名;其次在虚拟机或沙盒中下载,关闭共享功能并校验文件哈希;接着使用多引擎扫描工具检测恶意代码,分析行为日志;最后严格管理浏览器权限,禁用JavaScript和第三方插件,定期清除痕迹。 如果您尝试通过洋葱浏览器下载文件,但对来源和操作方式…

    2025年12月6日 软件教程
    000

发表回复

登录后才能评论
关注微信