Spring—AOP(简单介绍)


前言

AOP定义:面向切面编程
AOP的使用只存在于一些特定的场合(具有横切逻辑的应用场合),横切逻辑这个解释可能比较抽象,咱们说得再具体一点,AOP可以用于事务管理,日志管理,性能监测,权限校验等地方
Spring的AOP是通过代理模式实现的。实际生活中常见的代理场景如:房屋中介、黄牛售卖火车票。
代理分为静态代理和动态代理,Spring使用动态代理来完成AOP。
生成动态代理的两种方式:JDK动态代理(只支持实现接口的类)、CGLIB动态代理


一、XML版实现AOP

通过一个简单的案例演示AOP

如下为了实现用户注册和管理员注册,需要开启事务、关闭事务等大量重复的代码

public class UserServiceImpl implements IUserService {
    public void save() {
        try {
            System.out.println("开启事务");
            System.out.println("用户注册");
            System.out.println("提交事务");
        }catch (Exception e){
            e.printStackTrace();
            System.out.println("回滚事务");
        }finally {
            System.out.println("关闭事务");
        }
    }
}
public class DeptServiceImpl implements IDeptService{
    public void save() {
        try {
            System.out.println("开启事务");
            System.out.println("管理员注册");
            System.out.println("提交事务");
        }catch (Exception e){
            e.printStackTrace();
            System.out.println("回滚事务");
        }finally {
            System.out.println("关闭事务");
        }
    }
}

使用AOP简化上面的代码

将事务代码抽取成一个事务方法的类

public class TxManager {

    public void start(){
        System.out.println("开启事务");
    }

    public void conmit(){
        System.out.println("提交事务");
    }

    public void rollback(Throwable throwable){
        System.out.println("回滚事务,错误信息是:"+throwable);
    }

    public void close(){
        System.out.println("关闭事务");
    }
}

业务类只留下自己特有的方法即可

public class UserServiceImpl implements IUserService {
    public void save() {
        System.out.println("用户注册");
    }
}

配置SpringTest-Context.xml

如果有2个以上的切面增强,Spring可能会造成增强乱序

<bean id="userService" class="cn.itsource._03_aop.service.impl.UserServiceImpl"/>

    <bean id="txManager" class="cn.itsource._03_aop.TxManager"/>

    <!--开启aop-->
    <aop:config>
        <!--pointcut:切点,id:随便取,expression:要切的方法-->
        <!--第一个*表示所有的返回值
        第二个*表示所有的以I开头,service结尾的类
        两个..表示所有的参数,不同形参的save-->
        <aop:pointcut id="userPoincut" expression="execution(* cn.itsource._03_aop.service.I*Service.save(..))"/>

        <!--aspect:切面-->
        <aop:aspect ref="txManager">
            <!--在业务代码之前-->
            <aop:before method="start" pointcut-ref="userPoincut"/>
            <!--在业务代码之后-->
            <aop:after-returning method="conmit" pointcut-ref="userPoincut"/>
            <!--业务代码出现异常后 throwing:配置错误信息-->
            <aop:after-throwing method="rollback" pointcut-ref="userPoincut" throwing="throwable"/>
            <!--最终执行-->
            <aop:after method="close" pointcut-ref="userPoincut"/>
            
    </aop:config>

编写配置测试类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration
public class SpringTest {

    @Autowired
    private IUserService service;

    @Test
    public void test(){
        service.save();
    }
}

使用环绕增强

环绕增强是我们自己来控制切面的顺序,就不会出现乱序

public class TxManager {
    /**
     *环绕增强
     * @param joinPoint
     *
     * ProceedingJoinPoint能获取业务类
     */
    public void around(ProceedingJoinPoint joinPoint){
        try {
            start();
            //反射机制获取业务方法
            joinPoint.proceed();
            conmit();
        }catch (Throwable e){
            rollback(e);
        }finally {
            close();
        }
    }
}

配置xml

   <!--开启aop-->
    <aop:config>
        <!--pointcut:切点,id:随便取,expression:要切的方法-->
        <!--第一个*表示所有的返回值
        第二个*表示所有的以I开头,service结尾的类
        两个..表示所有的参数,不同形参的save-->
        <aop:pointcut id="userPoincut" expression="execution(* cn.itsource._03_aop.service.I*Service.save(..))"/>

        <!--aspect:切面-->
        <aop:aspect ref="txManager">
 
          <!--around:环绕切面,不会发生顺序问题。上面的配置多个切面方法spring可能发生顺序问题-->
          <aop:around method="around" pointcut-ref="userPoincut"/>

        </aop:aspect>
    </aop:config>

测试方法和上面的一样,但是需要注意,环绕增强和普通顺序增强不能同时出现,会重复执行

二、注解版实现AOP

实现类

@Service
public class UserServiceImpl implements IUserService {
    public void save() {
        System.out.println("用户注册");
    }
}

配置SpringTest-Context.xml

    <!--开启注解扫描-->
    <context:component-scan base-package="cn.itsource._04_aopanno"/>
    <!--支持aop注解-->
    <aop:aspectj-autoproxy/>

事务类,普通顺序增强的注解实现,这种方式也是可能会乱序

@Component  //声明bean
@Aspect    //声明该类是切面类
public class TxManager {

    //配置切点,通过一个空方法加注解实现,注解里的配置和xml的一样
    @Pointcut("execution(* cn.itsource._04_aopanno.service.I*Service.save(..))")
    public void poincut(){}

    @Before("poincut()")
    public void start(){
        System.out.println("开启事务");
    }

    @AfterReturning("poincut()")
    public void conmit(){
        System.out.println("提交事务");
    }

    @AfterThrowing(pointcut = "poincut()",throwing = "throwable")
    public void rollback(Throwable throwable){
        System.out.println("回滚事务,错误信息是:"+throwable);
    }

    @After("poincut()")
    public void close(){
        System.out.println("关闭事务");
    }
}

环绕增强,由于自己配置增强的顺序,所以不会出现乱序

@Component  //声明bean
@Aspect    //声明该类是切面类
public class TxManager {

    //配置切点,通过一个空方法加注解实现,注解里的配置和xml的一样
    @Pointcut("execution(* cn.itsource._04_aopanno.service.I*Service.save(..))")
    public void poincut(){}

    /**
     * @param joinPoint
     * ProceedingJoinPoint能获取业务类
     */
    @Around("poincut()")
    public void around(ProceedingJoinPoint joinPoint){
        try {
            start();
            //反射机制获取业务方法
            joinPoint.proceed();
            conmit();
        }catch (Throwable e){
            rollback(e);
        }finally {
            close();
        }
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spring AOPSpring框架的一部分,主要用于在应用程序的不同部分之间进行面向切面编程(AOP)。AOP是一种编程范式,可以将应用程序的关注分离,使应用程序更具可重用性和可维护性。 Spring AOP使用代理模式在运行时织入额外的功能,可以在方法调用前后、异常处理等方面添加额外的逻辑。使用Spring AOP可以将应用程序的关注(如事务处理、日志记录等)分离出来,从而提高应用程序的可维护性和可重用性。 Spring AOP支持使用AspectJ注解和XML配置两种方式来定义切面。Spring AOP还提供了一个强大的表达式语言(Spring Expression Language,简称SpEL),用于在运行时动态地匹配连接。 总的来Spring AOP是一个非常强大的AOP框架,可以帮助我们更好的管理应用程序的关注,提高应用程序的可维护性和可重用性。 ### 回答2: Spring AOP(面向切面编程)是Spring框架的一个核心模块,用于实现跨应用程序模块的关注分离。它通过在系统中将横切关注从业务逻辑中分离出来,实现了代码的重用性、可维护性和扩展性。 Spring AOP基于代理模式,通过动态代理技术在运行时将通用的横切关注织入目标对象中,从而实现了对目标对象方法的拦截和增强。Spring AOP主要通过使用切面、切、通知和连接等概念来实现。 - 切面(Aspect):切面是一个模块,它包含了一组与横切关注相关的通知和切。通常,一个应用程序由多个切面组成,每个切面负责某个特定的关注。比如,日志记录、性能监控等都可以作为切面。 - 切(Pointcut):切定义了在哪些目标对象方法上应用通知。它使用表达式来匹配目标对象中的方法。比如,通过使用通配符等方式,我们可以定义一个切来匹配所有的Service层方法。 - 通知(Advice):通知定义了在切上执行的行为。Spring AOP提供了五种类型的通知:前置通知(Before)、后置通知(AfterReturning)、异常通知(AfterThrowing)、最终通知(After)和环绕通知(Around)。通过在切面中配置不同类型的通知,我们可以在目标对象方法的不同执行阶段执行相应的操作。 - 连接(Joinpoint):连接是在目标对象中匹配到切的特定位置,比如方法的调用、方法的返回等。在连接上,可以执行相应的通知。 Spring AOP的主要优势是它能够将关注分离,使开发人员更关注业务逻辑而不是横切关注的实现。通过使用Spring AOP,我们可以实现日志记录、事务管理、性能监控等与业务逻辑无关的功能,增加了代码的可维护性和可重用性。此外,Spring AOP还提供了灵活的配置方式,可以使用XML配置或基于注解的方式来定义切面和通知,使代码更加简洁和易于理解。 总之,Spring AOPSpring框架中实现面向切面编程的核心模块,通过在运行时动态生成代理对象,实现了对目标对象方法的拦截和增强。它通过将通用的横切关注从业务逻辑中抽离出来,提高了代码的可维护性和可重用性。 ### 回答3: Spring AOP(Aspect-Oriented Programming)是Spring框架中的一个重要模块,用于实现面向切面编程的功能。它通过在不修改原有代码的情况下,将横切关注(如日志记录、安全检查、事务管理等)从业务逻辑中分离出来,从而提高系统的可维护性和可重用性。 Spring AOP的核心原理是基于动态代理技术,通过在运行时动态生成代理对象来实现AOP功能。它主要由以下几个核心概念组成: 1. 切面(Aspect):是对一个特定横切关注的抽象描述,它包含了一组通知(Advice)和一个切(Pointcut)。通知定义了在切处执行的具体操作,切定义了在应用程序中哪些位置应该触发通知。 2. 连接(Join point):是在程序运行过程中,符合切定义的特定位置,如方法调用、异常抛出等。Spring AOP仅支持方法级别的切。 3. 通知(Advice):是在连接处执行的具体操作,包括前置通知(Before)、后置通知(After)、返回通知(AfterReturning)、异常通知(AfterThrowing)和环绕通知(Around)等。它们分别在连接的不同位置执行,实现不同的横切关注功能。 4. 切表达式(Pointcut Expression):是一个表达式,用于匹配特定的切位置。Spring AOP支持使用AspectJ风格的切表达式语法。 Spring AOP的应用步骤如下: 1. 定义切面类,包含通知和切的定义。 2. 配置切面,通过XML配置或基于注解的方式将切面配置到Spring容器中。 3. 定义目标类,即需要被代理的类。 4. 配置代理,通过XML配置或基于注解的方式将目标类与切面关联起来。 5. 启动Spring容器,自动完成代理的实例化和连接的绑定。 6. 运行程序,当连接满足切定义时,代理会根据定义的通知类型执行相应的操作。 总之,Spring AOP是一种强大的横切关注分离工具,它通过代理机制实现对目标类的增强,使得系统的关注分离更加灵活和简单

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值