Spring-AOP详解

本文介绍了AOP(面向切面编程)的概念,强调其如何解决对象间重复代码的问题,通过自定义注解、切面和切点实现业务逻辑与核心代码的分离,提高代码复用性和可维护性。
摘要由CSDN通过智能技术生成

文章目录

        


前言

        当需要为多个不具有继承关系的对象引入一个公共的行为,例如:日志、安全检测等,起初我们只有在每个对象里面引用这些公共的行为,但是这样做会导致程序出现大量的重复代码,不便于维护。 


一、AOP切面是什么?

        前言中,引用了书上的一段话,读完之后,脑子里产生了一个问题,对于出现大量的重复代码,不便于维护的问题,我们也可以写一个公共方法,来给对象调用就行了,那么为什么还要使用切面呢?

        要解释这个问题,首先需要知道切面的关注点在哪里:切面所关注方向是横向的,而传统OOP是纵向的。怎么理解这段话?

        

       看下这张图,1-2-3-4这个步骤,是程序的业务代码,一步一步的向下执行;而A-B-C这个步骤,就是切面的逻辑,它和业务代码是分开的,不会影响业务代码的执行;如果使用引入公共方法的方式,那么必然会将其直接添加到业务代码中,这就使得业务代码和非业务代码耦合了。因此,总结下切面的优点:

  • 解耦,这个是最大的优点
  • 不影响业务代码,其实就是解耦的作用
  • 复用性好,一般切面都会自定义注解,这个注解可以用在很多地方,但是公共方法的模式也可以使用在很多地方,这个优势不算是很明显,但是比较简洁是真的。

二、实战

1.自定义注解

import java.lang.annotation.*;
 
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Log {
    //参数类
    Class< ? extends ConvertParam> convert();
 
    //操作类型
    BusinessType businessType() default BusinessType.OTHER;
 
}

2.定义切面、切点

@Aspect
@Component
public class LogAspect {
 
    @Resource
    HttpServletRequest httpServletRequest;
 
    /**
     * 定义切面
     */
    @Pointcut("@annotation(com.cart.cartservice.Log.annotation.Log)")
    public void section() {
    }
 
    /**
     * 环绕切点
     *
     * @param
     * @return result
     */
    @Around("section()")
    public Object log(ProceedingJoinPoint joinPoint) throws Throwable {
        //注解修饰的方法的返回结果
        Object result = joinPoint.proceed();
        //切面逻辑,收集日志什么的
        handleLog(joinPoint, result);
        //这个别忘记了
        return result;
    }
 
    @SneakyThrows
    private void handleLog(ProceedingJoinPoint joinPoint, Object result) {
        //标准三部曲,拿取注解
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();
        Log logAnnotation = method.getAnnotation(Log.class);
        
        //获取方法参数
        Object[] args = joinPoint.getArgs();
        //实现切面的逻辑。例如:收集日志等
    }
 
}

3. 使用注解

    @Log(convert = IntegerConvertParamImpl.class,businessType = BusinessType.QUERY)
    @GetMapping("{id}")
    public ResponseEntity<Cart> queryById(@PathVariable("id") Integer id) {
        return ResponseEntity.ok(this.cartService.queryById(id));
    }


总结

提示:这里对文章进行总结:
例如:以上就是今天要讲的内容,本文仅仅简单介绍了pandas的使用,而pandas提供了大量能使我们快速便捷地处理数据的函数和方法。

  • 15
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

何怀逸

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值