简介

1、什么叫做面向切面编程?

  

  概念:把一个个的横切关注点(某种业务的实现代码)放到某个模块中去,称之为切面。每个切面影响业务的一种功能,切面的目的就是为了功能增强,将需要增强的方法做成切面,实现对业务的增强,就是面向切面编程。

  目的:将与业务本身无关,却被业务模块所共同调用的功能代码封装成切面,以减少系统的重复代码,降低耦合,提高可扩展性。

  优势:把多个方法前/后的共同代码抽离出来,使用动态代理机制来控制,先执行抽离出来的代码,再执行每一个真实方法.

2、Spring中的AOP使用动态代理来实现:

  如果一个类实现了接口,那么spring就使用JDK的动态代理完成AOP;

  如果一个类没有实现接口,那么spring就是用cglib完成AOP。

3、AOP的一些基本概念

  Joinpoint:连接点,被拦截到需要被增强的方法。去哪里做增强

  Pointcut:切入点,哪些包中的哪些类中的哪些方法,可认为是连接点的集合。去哪些地方做增强

  Advice:增强,当拦截到Joinpoint之后,在方法执行的什么时机(when)做什么样(what)的增强。

  Aspect:切面,Pointcut+Advice,去哪些地方+在什么时候+做什么增强

  Weaving:织入,把Advice加到Target上之后,创建出Proxy对象的过程。

切入点语法

1、execution(<访问修饰符>?<返回值类型><声明类型>?<方法名>(<参数名>)<异常名>)

  ?表示出现0次或1次

  通配符:

     * :匹配任何部分,只能表示一个单词

     .. : 可用于全限定名中和方法参数中,分别表示子包和0到N个参数

2、实例介绍:

   

 

 

 XML方式实现AOP

1、导入依赖

  spring-aop  spring-aspectj

2、命名空间

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       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
">
</beans>

3、配置AOP

    <!--配置AOP-->
    <aop:config>
        <!--配置切入点pointcut
            哪些包里的哪些类需要被增强
        -->
        <aop:pointcut id="pointcut" expression="execution( * com.test.class03_AOP_xml.service.IUserService.*(..))"/>
        <!--配置切面Aspect
            Aspect=pointcut+Advice
        -->
        <aop:aspect ref="txManager">
            <!--前置增强-->
            <aop:before method="begin" pointcut-ref="pointcut"/>
            <!--后置增强-->
            <aop:after-returning method="commit" pointcut-ref="pointcut"/>
            <!--异常增强-->
            <aop:after-throwing method="rollback" pointcut-ref="pointcut"/>
            <!--最终增强-->
            <aop:after method="destroy" pointcut-ref="pointcut"/>
        </aop:aspect>
    </aop:config>

4、AOP中几个不同的增强时机:

  aop:before(前置增强):在方法执行之前执行增强;

  aop:after-returning(后置增强):在方法正常执行完成之后执行增强;

  aop:after-throwing(异常增强):在方法抛出异常退出时执行增强;

  aop:after(最终增强):在方法执行之后执行,相当于在finally里面执行;可以通过配置throwing来获得拦截到的异常信息

  aop:around(环绕增强):最强大的一种增强类型。

    环绕增强可以在方法调用前后完成自定义的行为,环绕增强有两个要求

    1、方法要返回一个Object(返回的结果)

    2、方法的第一个参数是ProceedingJoinPoint(可以继续向下传递的切入点)

<!--环绕增强-->
<aop:around method="around" pointcut-ref="pointcut"/>

5、代码示例

public class TxManager {
    public void begin() {
        System.out.println("开启事务###");
    }
    public void commit() {
        System.out.println("提交事务++++++");
    }
    public void rollback() {
        System.out.println("回滚事务....");
    }
    public void destroy() {
        System.out.println("释放资源……………………");
    }
    //环绕增强代码示例
    public Object around(ProceedingJoinPoint pjp){
        Object obj=null;
        try {
            System.out.println("开启事务###");
            obj=pjp.proceed();
            System.out.println("提交事务++++++");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
            System.out.println("回滚事务....");
        }finally {
            System.out.println("释放资源……………………");
        }
        return obj;
    }
}

6、获取增强的参数

  1.在增强方法中获取异常的信息。

    <aop:after-throwing>的标签中添加throwing=“ex”的属性

    增强方法rollback中,添加形式参数:Exception ex。

    则形参ex中就自动注入了异常对象。 注意:throwing属性的值,必须与方法中形参的名字相同

  2.在增强方法中,获取被增强方法的信息

    Spring AOP提供org.aspectj.lang.JoinPoint类作为增强方法的第一个参数。

    JoinPoint :提供访问当前被增强方法的真实对象、代理对象、方法参数等数据。

    ProceedingJoinPoint:JinPoint子类,只用于环绕增强中,可以处理被增强方法。

      jp.getThis():获取代理对象

      jp.getTarget():获取目标对象

      jp.getArgs():获取被增强方法的参数

      jp.getSignature():获取被增强方法的参数

注解方式实现AOP

1、命名空间

2、添加注解解析器<aop:aspect-autoproxy/>

3、@Aspect 切面,下面的注解都在切面里配置

  @before

  @AfterReturning

  @AfterThrowing

  @After

  @Around

 4、代码示例

@Aspect
public class TxManager {
    @Pointcut("execution( * com.test.class04_AOP_Anno.service.IUserService.*(..))")
    public void tt(){}
    @Before("tt()")
    public void begin() {
        System.out.println("开启事务###");
    }
    @AfterReturning("tt()")
    public void commit() {
        System.out.println("提交事务++++++");
    }
    @AfterThrowing(value="tt()",throwing = "ex")
    public void rollback() {
        System.out.println("回滚事务....");
    }
    @After("tt()")
    public void destroy() {
        System.out.println("释放资源……………………");
    }
    @Around("tt()")
    public Object around(ProceedingJoinPoint pjp){
        Object obj=null;
        try {
            System.out.println("开启事务###");
            obj=pjp.proceed();
            System.out.println("提交事务++++++");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
            System.out.println("回滚事务....");
        }finally {
            System.out.println("释放资源……………………");
        }
        return obj;
    }
}

 

posted on 2019-09-08 20:14  幸福的小耗子  阅读(489)  评论(0编辑  收藏  举报