AOP面向切面编程

AOP简介

AOP(Aspect oriented Programming)面向切面编程,就是面向特定的方法编程,将方法比作一个一个的切面,可以向指定的方法执行前/后执行自己的逻辑。如统一获取方法的时间。

应用场景:
记录操作日志、权限控制、事务管理

优势:
代码无侵入(无需在原先的代码上修改)、减少重复代码、提高开发效率、维护方便

原理:

动态代理,运行时会注入一个动态代理对象(实现了注入的接口),动态代理对象会先后执行切片前的方法、调用方法、切片后的方法。
在动态代理时的userMapper:
在这里插入图片描述

核心概念

JoinPoint(连接点):能够被控制的方法,在spring中指方法的运行,含有方法执行时的相关信息。
Advice(通知):指重复、相同的功能,即共性功能。
PointCut(切入点):匹配连接的条件,实际被AOP控制的方法,通知只有在切入点方法执行时被应用。
Aspect(切面):描述通知和切入点的对应关系
target(目标对象):通知所应用的对象。

使用方式:
1.创建切面类 标记注解@Aspect @Component
2.创建切面方法参数为ProceedingJoinPoint(连接点),在方法上加入@Around注解表明该切面在哪个方法调用时执行。

切入点表达式:
  @Around("execution(* com.yi.springbootwebquickstart.controller.*.*(..))")  
  第一个*表示任意返回值,第二个*表示controller包下的任意类,第三个*表示类中的任意方法 
  根据自定义注解匹配:
  @Around(@annotation(com.yi.springbootwebquickstart.aop.mylog))
@Component
@Aspect
@Slf4j
public class TimeAspect {
    @Around("execution(* com.yi.springbootwebquickstart.controller.*.*(..))")
    public void recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
        long begin = System.currentTimeMillis();

        Object result = joinPoint.proceed();
        long end = System.currentTimeMillis();

        log.info("{} cost time:"+(end-begin),joinPoint.getSignature());
    }
}

通知

1.通知类型

@Around 环绕通知,在目标方法执行前后执行
@Before 前置通知,在目标方法执行前执行
@After 后置通知,在目标方法执行后执行,有异常也会执行
@AfterReturning 返回后通知,在目标方法返回后执行,有异常不会执行
@AfterThrowing 异常后通知,在目标方法发生异常后执行


@Component
@Aspect
@Slf4j
public class TimeAspect {
	//定义一个方法,将相同的切入点表达式抽出出来
    @Pointcut("execution(* com.yi.springbootwebquickstart.controller.*.*(..))") 
    public void pt(){}
    @Around("pt()")
    public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
        long begin = System.currentTimeMillis();

        Object result = joinPoint.proceed();
        long end = System.currentTimeMillis();

        log.info("{} cost time:"+(end-begin),joinPoint.getSignature());
        return result;
    }

    @Before("pt()")
    public void before(){
        System.out.println("before");
    }

    @After("pt()")
    public void after(){
        System.out.println("after");
    }
    @AfterReturning("pt()")
    public void afterReturning(){
        System.out.println("afterReturning");
    }
    
    @AfterThrowing("pt()")
    public void afterThrowing(){
        System.out.println("afterThrowing");
    }
}

2.通知顺序

默认是按照类名以环绕顺序来执行,字母越靠前:通知前方法越先执行,通知后方法越后执行。
在这里插入图片描述
可以加入在类名上加入@Order(int) 来控制执行顺序,数字越小通知前方法越先执行,通知后方法越后执行。

连接点:可以被AOP控制的方法

在Spring中特指方法的运行,可以通过连接点获取方法执行时的相关信息,如类名、方法名、方法参数等。
@Around通知只能用ProceedingJoinPoint,其他通知只能用JointPoint

log.info("目标方法的类名:{}",joinPoint.getTarget().getClass().getName());
log.info("目标方法签名:{}",joinPoint.getSignature());
log.info("目标方法方法名:{}",joinPoint.getSignature().getName());
log.info("运行参数:{}",joinPoint.getArgs());
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的AOP面向切面编程的测试代码示例,使用Spring框架实现: 首先,创建一个切面类 `LoggingAspect`,用于定义切面逻辑: ```java import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; @Aspect @Component public class LoggingAspect { @Before("execution(* com.example.service.*.*(..))") public void beforeAdvice(JoinPoint joinPoint) { System.out.println("Before method: " + joinPoint.getSignature().getName()); } @After("execution(* com.example.service.*.*(..))") public void afterAdvice(JoinPoint joinPoint) { System.out.println("After method: " + joinPoint.getSignature().getName()); } } ``` 然后,创建一个测试服务类 `UserService`,用于演示AOP的应用: ```java import org.springframework.stereotype.Service; @Service public class UserService { public void createUser(String username) { System.out.println("Creating user: " + username); } public void deleteUser(String username) { System.out.println("Deleting user: " + username); } } ``` 最后,创建一个Spring Boot应用程序,并在启动类中进行配置: ```java import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.context.annotation.EnableAspectJAutoProxy; @SpringBootApplication @EnableAspectJAutoProxy public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); UserService userService = context.getBean(UserService.class); userService.createUser("John"); userService.deleteUser("John"); } } ``` 在上述示例中,`LoggingAspect` 切面类使用 `@Before` 和 `@After` 注解分别定义了在目标方法执行前和执行后的逻辑。切面逻辑会应用于 `UserService` 类中的所有方法。 当运行应用程序时,可以看到切面逻辑在方法执行前和执行后打印了相应的日志消息。 这是一个简单的AOP面向切面编程的示例,你可以根据实际需求进行更复杂的切面逻辑定义和应用。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值