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

【Spring】AOP在实际项目中的运用

一、AOP 的核心思想:注解驱动

通过自定义注解标记需要增强的方法,切面逻辑根据注解动态增强代码。业务代码无需感知具体逻辑,只需关注自身职责


二、电商通用场景示例

1. 接口防重提交(幂等性控制)

定义注解:标记需要幂等控制的接口。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Idempotent {
    String key() default "";  // 支持动态生成幂等键
    int expire() default 5;    // 默认锁定时长5秒
}

切面逻辑:统一处理幂等键的生成与校验。

@Aspect
public class IdempotentAspect {
    @Around("@annotation(idempotent)")
    public Object checkIdempotent(ProceedingJoinPoint joinPoint, Idempotent idempotent) throws Throwable {
        String key = generateKey(joinPoint, idempotent); // 动态生成幂等键
        if (redis.setIfAbsent(key, "1", idempotent.expire(), TimeUnit.SECONDS)) {
            return joinPoint.proceed(); // 执行业务逻辑
        } else {
            throw new BusinessException("请勿重复请求!");
        }
    }
}

业务代码:只需添加注解,无需编写防重逻辑。

@Idempotent(key = "#order.userId + '-' + #order.productId") // 动态生成键
public void createOrder(Order order) {
    // 业务逻辑(无需关心防重)
}

2. 接口权限校验(动态鉴权)

定义注解:标记需要鉴权的接口。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface RequirePermission {
    String value();  // 权限标识(如 "order:create")
}

切面逻辑:统一从上下文获取用户权限并校验。

@Aspect
public class PermissionAspect {
    @Before("@annotation(RequirePermission)")
    public void checkPermission(JoinPoint joinPoint) {
        RequirePermission annotation = getAnnotation(joinPoint);
        if (!currentUser.hasPermission(annotation.value())) {
            throw new BusinessException("无权限操作!");
        }
    }
}

业务代码:仅用注解声明所需权限。

@RequirePermission("order:create")
public void createOrder(Order order) {
    // 业务逻辑(无需关心鉴权)
}

3. 热点数据缓存(自动缓存)

定义注解:标记需要缓存的方法。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface CacheResult {
    String key();         // 缓存键(支持SpEL表达式)
    int expire() default 3600; // 缓存时间(秒)
}

切面逻辑:统一处理缓存的读取与写入。

@Aspect
public class CacheAspect {
    @Around("@annotation(cacheResult)")
    public Object cacheResult(ProceedingJoinPoint joinPoint, CacheResult cacheResult) throws Throwable {
        String key = parseKey(joinPoint, cacheResult); // 解析SpEL表达式生成键
        Object value = redis.get(key);
        if (value != null) return value;
        value = joinPoint.proceed(); // 执行业务逻辑
        redis.set(key, value, cacheResult.expire(), TimeUnit.SECONDS);
        return value;
    }
}

业务代码:只需关注数据查询逻辑。

@CacheResult(key = "'product:' + #productId") // 自动缓存结果
public Product getProductDetail(String productId) {
    return productDao.findById(productId);
}

4. 接口限流(动态阈值)

定义注解:标记需要限流的接口。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface RateLimit {
    int permitsPerSecond() default 100; // 每秒允许请求数
}

切面逻辑:统一接入限流算法(如令牌桶)。

@Aspect
public class RateLimitAspect {
    private final Map<String, RateLimiter> limiters = new ConcurrentHashMap<>();
    
    @Around("@annotation(rateLimit)")
    public Object limit(ProceedingJoinPoint joinPoint, RateLimit rateLimit) throws Throwable {
        String key = generateKey(joinPoint); // 按方法名生成限流键
        RateLimiter limiter = limiters.computeIfAbsent(key, 
            k -> RateLimiter.create(rateLimit.permitsPerSecond()));
        if (limiter.tryAcquire()) {
            return joinPoint.proceed();
        } else {
            throw new BusinessException("请求过于频繁!");
        }
    }
}

业务代码:仅用注解声明限流阈值。

@RateLimit(permitsPerSecond = 50) // 每秒最多50次请求
public void submitOrder(Order order) {
    // 业务逻辑(无需关心限流)
}

三、AOP 的核心优势

  1. 解耦性:业务代码不耦合横切逻辑(如防重、鉴权)。
  2. 复用性:同一套切面逻辑可被多个注解复用。
  3. 动态性:通过 SpEL 表达式实现注解参数的动态计算。
  4. 可维护性:修改横切逻辑时只需调整切面,无需改动业务代码。

四、对比直接在业务代码中实现

场景AOP + 注解业务代码硬编码
防重提交通过 @Idempotent 注解自动处理每个方法中编写 Redis 操作与异常处理
权限校验通过 @RequirePermission 注解自动拦截在每个方法开头调用权限服务
缓存逻辑通过 @CacheResult 注解自动缓存手动查询缓存、处理未命中与回填逻辑
限流控制通过 @RateLimit 注解自动限流在每个方法中集成限流算法与异常处理

五、总结

通过 自定义注解 + 动态切面,AOP 在电商项目中能实现:
快速接入通用能力:新接口只需添加注解即可获得防重、鉴权等功能。
统一逻辑管理:所有缓存、限流逻辑集中在切面中,避免代码分散。
灵活调整策略:修改限流阈值或缓存时间只需调整注解参数,无需修改业务代码。

这种设计模式特别适合中大型项目,能显著提升开发效率与系统可维护性。

相关文章:

  • 程序化广告行业(6/89):现状、未来与核心要点剖析
  • 使用Process Explorer、Dependency Walker和PE信息查看工具快速排查dll动态库因库与库版本不一致导致的加载失败问题
  • 支持selenium的chrome driver更新到134.0.6998.88
  • SpringBoot开发——整合SpringReport开源报表工具
  • 如何在 React 中使用 CSS-in-JS?
  • 【Go每日一练】构建一个简单的用户信息管理系统
  • Smart contract -- 自毁合约
  • 【动手实验】TCP 连接的建立与关闭抓包分析
  • 【Pandas】pandas Series asfreq
  • 【Axure视频教程】中继器表格——控制开关按钮
  • spark常见的submit参数
  • HeidiSQL 12.0 64位便携版使用指南:从下载到数据库管理的完整步骤
  • 个人学习编程(3-11) 刷题
  • 《哪吒2》中的“家”本质:中国式亲情的三重镜像与觉醒
  • Spring
  • SPFA算法——负权图且没有负环
  • Python基于深度学习的电影评论情感分析可视化系统(全新升级版)【附源码、参考文档】
  • 【BUG】类文件具有错误的版本 61.0, 应为 52.0,请删除该文件或确保该文件位于正确的类路径子目录中。
  • ubuntu安装与卸载
  • python机器学习tensorflow库安装与使用
  • 美乌基金协议:美国搞了一套可在资源富集地区复刻的商业模式
  • 保证断电、碰撞等事故中车门系统能够开启!隐藏式门把手将迎来强制性国家标准
  • 人民日报整版调查:中小学春秋假,如何放得好推得开?
  • 欧盟委员会计划对950亿欧元美国进口产品采取反制措施
  • 洞天寻隐·学林纪丨玉洞桃源:仇英青绿山水画中的洞天与身体
  • 人民日报钟声:中方维护自身发展利益的决心不会改变