一:Schema-based (前置后置通知)

1.新建通知类

  1.1 新建前置通知类

    1.1.1 arg0: 切点方法对象 Method 对象

    1.1.2 arg1: 切点方法参数

    1.1.3 arg2:切点在哪个对象中

  • 前置通知类代码
public class MyBeforeAdvice implements MethodBeforeAdvice {
  @Override
  public void before(Method arg0, Object[] arg1, Objectarg2) throws Throwable {
  System.out.println("执行前置通知");
  }
}

1.2 新建后置通知类

  1.2.1 arg0: 切点方法返回值

  1.2.2 arg1:切点方法对象

  1.2.3 arg2:切点方法参数

  1.2.4 arg3:切点方法所在类的对象

  • 后置通知类代码
public class MyAfterAdvice implements AfterReturningAdvice {

@Override
public void afterReturning(Object arg0, Method arg1,Object[] arg2, Object arg3) throws Throwable {
  System.out.println("执行后置通知");
  }
}

1.3配置 spring 配置文件

  1.3.1 引入 aop 命名空间

  1.3.2 配置通知类的<bean>

  1.3.3 配置切面

  1.3.4 * 通配符,匹配任意方法名,任意类名,任意一级包名

  1.3.5 如果希望匹配任意方法参数     (..)

  • spring配置文件
<?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/sc hema/beans
http://www.springframework.org/schema/beans/spring-be ans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop. xsd">

<!-- 配置通知类对象,在切面中引入 -->
<bean id="mybefore" class="com.advice.MyBeforeAdvice"></bean>
<bean id="myafter" class="com.advice.MyAfterAdvice"></bean>

<!-- 配置切面 -->
<aop:config>
<!-- 配置切点 -->
<aop:pointcutexpression="execution(*com.test.Demo.demo2())"id="mypoint"/>
<!-- 通知 -->
<aop:advisor advice-ref="mybefore"pointcut-ref="mypoint"/>
<aop:advisor advice-ref="myafter"pointcut-ref="mypoint"/>
</aop:config>

<!-- 配置 Demo 类,测试使用 -->
<bean id="demo" class="com.test.Demo"></bean>

</beans>
  • 测试类代码
public class Test {

public static void main(String[] args) {
     ApplicationContext ac = new 
     ClassPathXmlApplicationContext("applicationContext.xm l");
     Demo demo = ac.getBean("demo",Demo.class);
     demo.demo1();
     demo.demo2();
     demo.demo3();
   }
}

二:Schema-based ( 异常通知)

1 新建一个类实现 throwsAdvice 接口

  1.1 必须自己写方法,且必须叫 afterThrowing

  1.2 有两种参数方式

    1.2.1 必须是 1 个或 4 个

    1.3 异常类型要与切点报的异常类型一致

  • 异常通知类
public class MyThrow implements ThrowsAdvice{

public void afterThrowing(Exception ex) throws Throwable {

  System.out.println("执行异常通过-schema-base 方式");

   }
}
  • 配置文件
<bean id="mythrow" class="com.advice.MyThrow"></bean>

<!-- 配置切面 -->
<aop:config>
<!-- 配置切点 -->
<aop:pointcut expression="execution(*com.test.Demo.demo1())"id="mypoint"/>
<!-- 通知 -->
<aop:advisor advice-ref="mythrow"pointcut-ref="mypoint" />
</aop:config>

<bean id="demo" class="com.test.Demo"></bean>

三:Schema-based (环绕通知)

1 把前置通知和后置通知都写到一个通知中,组成了环绕通知

2 实现步骤

 2.1 新建一个类实现 MethodInterceptor

  • 通知类
public class MyArround implements MethodInterceptor { @Override

  public Object invoke(MethodInvocation arg0) throws Throwable {

    System.out.println("环绕-前置");
    Object result = arg0.proceed();//放行,调用切点方式
    System.out.println("环绕-后置");
    return result;

  }
}
  • 配置文件
<bean id="myarround" class="com.advice.MyArround"></bean> 
<!-- 配置切面 -->
<aop:config>
<!-- 配置切点 -->
<aop:pointcut expression="execution(* com.test.Demo.demo1())"id="mypoint"/>
<!-- 通知 -->
<aop:advisor advice-ref="myarround" pointcut-ref="mypoint" />
</aop:config>
<!-- 配置 Demo 类,测试使用 -->
<bean id="demo" class="com.test.Demo"></bean>

四:AspectJ(前置后置环绕通知)

4 新建类,不用实现

  4.1 类中方法名任意

  • 通知类
public class MyAdvice {

  public void mybefore(String name1,int age1){

    System.out.println("前置"+name1 );

  }
  public void mybefore1(String name1){ 

    System.out.println("前置:"+name1);

  }
  public void myaftering(){

    System.out.println("后置 2");

  }
  public void myafter(){

    System.out.println("后置 1");

  }
  public void mythrow(){

    System.out.println("异常");

  }
  public Object myarround(ProceedingJoinPoint p) throws Throwable{

    System.out.println("执行环绕");

    System.out.println("环绕-前置");

    Object result = p.proceed();

    System.out.println("环绕后置");

    return result;
   }
}

4.2 配置 spring 配置文件

  4.2.1 <aop:after/> 后置通知,是否出现异常都执行

  4.2.2 <aop:after-returing/> 后置通知,只有当切点正确执行时执行

  4.2.3 <aop:after/> 和 <aop:after-returing/> 和<aop:after-throwing/>执行顺序和配置顺序有关

  4.2.4 execution() 括号不能扩上 args

  4.2.5 中间使用 and 不能使用&& 由 spring 把 and 解析成&&

  4.2.6 args(名称) 名称自定义的.顺序和 demo1(参数,参数)对应

  4.2.7 <aop:before/> arg-names=” 名 称 ” 名 称 来 源 于 expression=”” 中 args(),名称必须一样

    4.2.7.1 args() 有几个参数,arg-names 里面必须有几个参数

    4.2.7.2 arg-names=”” 里面名称必须和通知方法参数名对应

  • 配置文件
<!--配置切面-->
<aop:config>
<aop:aspect ref="myadvice">
<!--配置切点-->
<aop:pointcut expression="execution(* com.test.Demo.demo1(String,int)) and args(name1,age1)" id="mypoint"/>
<aop:pointcut expression="execution(* com.test.Demo.demo1(String)) and args(name1)" id="mypoint1"/>
<!--配置前置通知-->
<aop:before method="mybefore" pointcut-ref="mypoint" arg-names="name1,age1"/> 
<aop:before method="mybefore1" pointcut-ref="mypoint1" arg-names="name1"/> 
<!--配置后置通知-->
<aop:after method="myafter" pointcut-ref="mypoint"/>
<aop:after-returning method="myaftering" pointcut-ref="mypoint"/> <aop:after-throwing method="mythrow" pointcut-ref="mypoint"/>
<!--配置环绕通知-->
<aop:around method="myarround" pointcut-ref="mypoint"/>
</aop:aspect>
</aop:config>

五:AspectJ(异常通知)

5.1只有当切点报异常才能触发异常通知

5.2在 spring 中有 AspectJ 方式提供了异常通知的办法.

  5.2.1 如果希望通过 schema-base 实现需要按照特定的要求自己编写方法.

5.3实现步骤:

  5.3.1 新建类,在类写任意名称的方法

  • 通知类
public class MyThrowAdvice{

public void myexception(Exception e1){

    System.out.println("执行异常通知"+e1.getMessage());

  }
}

  5.3.2 在 spring 配置文件中配置

    5.3.2.1 <aop:aspect>的 ref 属性表示:方法在哪个类中.

    5.3.2.2 <aop: xxxx/> 表示什么通知

    5.3.2.3 method: 当触发这个通知时,调用哪个方法

    5.3.2.4 throwing: 异常对象名,必须和通知中方法参数名相同(可以不在通知中声明异常对象)

  • 配置文件
<bean id="mythrow" class="com.advice.MyThrowAdvice"></bean>
<!-- 配置切面 -->
<aop:config>
<aop:aspect ref="mythrow">
<!--配置切点-->
<aop:pointcut expression="execution(* com.test.Demo.demo1())" id="mypoint"/>
<!--配置异常通知-->
<aop:after-throwing method="myexception" pointcut-ref="mypoint" throwing="e1"/>
</aop:aspect>
</aop:config>
<!-- 配置 Demo 类,测试使用 -->
<bean id="demo" class="com.test.Demo"></bean>

For sharing , For emulating , For enterprising