Spring-AOP三种实现方式


一、什么是AOP?

1、面向切面编程(方面),利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

2、通俗描述:不通过修改源代码方式,在主干功能里面添加新功能

Aop在Spring中的作用

提供声明式事务;允许用户自定义切面

AOP中的专业术语

  • 连接点:类里面哪些方法可以被增强,这些方法称为连接点
  • 切入点:实际被真正增强的方法,称为切入点
  • 通知(增强):实际增强的逻辑部分称为通知(增强)
  • 切面:把通知应用到切入点的过程
  • 注意点:通知有多种类型,如前置通知、后置通知、环绕通知、异常通知、最终通知

二、实现方式(jdk动态代理)

1、方式一(通过 Spring API 实现)

前提:需要导入AOP依赖包:

<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
	<groupId>org.aspectj</groupId>
	<artifactId>aspectjweaver</artifactId>
	<version>1.9.4</version>
</dependency>

第一步:编写我们的业务接口和实现类

public interface UserService {

   public void add();

   public void delete();

   public void update();

   public void search();

}
public class UserServiceImpl implements UserService{

   @Override
   public void add() {
       System.out.println("增加用户");
  }

   @Override
   public void delete() {
       System.out.println("删除用户");
  }

   @Override
   public void update() {
       System.out.println("更新用户");
  }

   @Override
   public void search() {
       System.out.println("查询用户");
  }
}

第二步:然后去写我们的增强类 , 我们编写两个 , 一个前置增强 一个后置增强

public class Log implements MethodBeforeAdvice {

   //method : 要执行的目标对象的方法
   //objects : 被调用的方法的参数
   //Object : 目标对象
   @Override
   public void before(Method method, Object[] objects, Object o) throws Throwable {
       System.out.println( o.getClass().getName() + "的" + method.getName() + "方法被执行了");
  }
}
public class AfterLog implements AfterReturningAdvice {
   //returnValue 返回值
   //method被调用的方法
   //args 被调用的方法的对象的参数
   //target 被调用的目标对象
   @Override
   public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
       System.out.println("执行了" + target.getClass().getName()
       +"的"+method.getName()+"方法,"
       +"返回值:"+returnValue);
  }
}

第三步:最后去spring的文件中注册 , 并实现aop切入实现 , 注意导入约束 .

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:aop="http://www.springframework.org/schema/aop"
      xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop.xsd">

   <!--注册bean-->
   <bean id="userService" class="com.kuang.service.UserServiceImpl"/>
   <bean id="log" class="com.kuang.log.Log"/>
   <bean id="afterLog" class="com.kuang.log.AfterLog"/>

   <!--aop的配置-->
   <aop:config>
       <!--切入点 expression:表达式匹配要执行的方法-->
       <aop:pointcut id="pointcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
       <!--执行环绕; advice-ref执行方法 . pointcut-ref切入点-->
       <aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
       <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
   </aop:config>

</beans>

其中execution(* com.kuang.service.UserServiceImpl.*(…))

解释如下:

符号含义
execution()表达式的主体;
第一个”*“符号表示返回值的类型任意;
com.kuang.service.UserServiceImplAOP所切的服务的包名,即,我们的业务部分
包名后面的”.“表示当前包及子包
.*(…)表示任何方法名,括号表示参数,两个点表示任何参数类型

第四步:测试

public class MyTest {
   @Test
   public void test(){
       ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
       //注意这里调用的是接口而不是实现类
       UserService userService = context.getBean("userService",UserService.class);
       userService.search();
  }
}

2、方式二(自定义类实现AOP)

自定义类来实现Aop

目标业务类不变依旧是userServiceImpl

第一步 : 自定义一个切入类

public class DiyPointcut {

   public void before(){
       System.out.println("---------方法执行前---------");
  }
   public void after(){
       System.out.println("---------方法执行后---------");
  }
   
}

第二步:去spring中配置

<!--第二种方式自定义实现-->
 <!--注册bean-->
    <!--切入的目标类-->
    <bean id="userService" class="com.zhao.service.UserServiceImpl"></bean>
    <!--自定义的切入类即切面-->
    <bean id="diyCut" class="com.zhao.diy.DiyPointCut"></bean>

<!--aop的配置-->
<aop:config>
   <!--第二种方式:使用AOP的标签实现-->
   <aop:aspect ref="diyCut">
       <aop:pointcut id="diyPonitcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
       <aop:before pointcut-ref="diyPonitcut" method="before"/>
       <aop:after pointcut-ref="diyPonitcut" method="after"/>
   </aop:aspect>
</aop:config>

第三步:测试

public class MyTest {
   @Test
   public void test(){
       ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
       UserService userService = context.getBean("userService",UserService.class);
       userService.add();
  }
}

3、方式三(基于注解的方式进行实现)

第一步:编写一个注解实现的增强类

package com.zhao.diy;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;

@Aspect
public class AnnotationPointCut {

    @Before("execution(* com.zhao.service.UserServiceImpl.*(..))")
    public void before(){
        System.out.println("---------方法执行前---------");
    }

    @After("execution(* com.zhao.service.UserServiceImpl.*(..))")
    public void after(){
        System.out.println("---------方法执行后---------");
    }
    
    //相同切入点抽取
    @Pointcut(value = "execution(* com.zhao.service.UserServiceImpl.*(..))")
    public void pointDemo() {
    }

    @Around(value = "pointDemo()")
    public void around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("环绕前");
        System.out.println("签名:"+jp.getSignature());
        //执行目标方法proceed
        Object proceed = jp.proceed();
        System.out.println("环绕后");
    }



}

此外,在使用注解时,为了增强减少代码冗余,可以创建一个方法来替代execution表达式。如:

	//相同切入点抽取
    @Pointcut(value = "execution(* com.zhao.service.UserServiceImpl.*(..))")
    public void pointDemo() {
    }

    @Around(value = "pointDemo()")
    public void around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("环绕前");
        System.out.println("签名:"+jp.getSignature());
        //执行目标方法proceed
        Object proceed = jp.proceed();
        System.out.println("环绕后");
    }

第二步:在Spring配置文件中,注册bean,并增加支持注解的配置

  <!--注册bean-->
    <bean id="userServiceImpl" class="com.zhao.service.UserServiceImpl"></bean>

    <!--在Spring配置文件中,注册bean,并增加支持注解的配置-->
    <bean id="pointCut" class="com.zhao.diy.AnnotationPointCut"></bean>
    <!--开启自动注解-->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>


总结

Spring的Aop就是将公共的业务 (日志 , 安全等) 和领域业务结合起来 , 当执行领域业务时 , 将会把公共业务加进来 . 实现公共业务的重复利用 . 领域业务更纯粹 , 程序猿专注领域业务 , 其本质还是动态代理 。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值