Spring详解(四)

AOP 关键术语

1.target:目标类,需要被代理的类。例如:UserService

  2.Joinpoint(连接点):所谓连接点是指那些可能被拦截到的方法。例如:所有的方法

  3.PointCut 切入点:已经被增强的连接点。例如:addUser()

  4.advice 通知/增强,增强代码。例如:after、before

  5. Weaving(织入):是指把增强advice应用到目标对象target来创建新的代理对象proxy的 过程.

  6.proxy 代理类:通知+切入点

  7. Aspect(切面): 是切入点pointcut和通知advice的结合

  具体可以根据下面这张图来理解:


AOP 的通知类型

Spring按照通知Advice在目标类方法的连接点位置,可以分为5类

前置通知

org.springframework.aop.MethodBeforeAdvice

在目标方法执行前实施增强,比如上面例子的 before()方法

后置通知

org.springframework.aop.AfterReturningAdvice

在目标方法执行后实施增强,比如上面例子的 after()方法

环绕通知

org.aopalliance.intercept.MethodInterceptor

在目标方法执行前后实施增强

异常抛出通知

org.springframework.aop.ThrowsAdvice

在方法抛出异常后实施增强

引介通知

org.springframework.aop.IntroductionInterceptor

在目标类中添加一些新的方法和属性

我们只需要在Spring 的配置文件 applicationContext.xml 进行如下配置

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">

@Test

public void testAop(){

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

UserService useService = (UserService) context.getBean("userService");

useService.addUser(null);

}

 ①、 切入点表达式,一个完整的方法表示如下:

execution(modifiers-pattern? ref-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)

类修饰符 返回值 方法所在的包 方法名 方法抛出的异常

那么根据上面的对比,我们就很好理解:

execution(* com.my.aop.*.*(..))

选择方法 返回值任意 包 类名任意 方法名任意 参数任意

那么它表达的意思是 返回值任意,包名为 com.ys.aop 下的任意类名中的任意方法名,参数任意。

如果切入点表达式有多个不同目录呢?

表示匹配 com.my包下的,以 Service1结尾或者以Service2结尾的类的任意方法。

 AOP 切入点表达式支持多种形式的定义规则:

1、execution:匹配方法的执行(常用)

execution(public *.*(..))

2.within:匹配包或子包中的方法(了解)

within(com.my.aop..*)

3.this:匹配实现接口的代理对象中的方法(了解)

this(com.my.aop.user.UserDAO)

4.target:匹配实现接口的目标对象中的方法(了解)

target(com.my.aop.user.UserDAO)

5.args:匹配参数格式符合标准的方法(了解)

args(int,int)

6.bean(id) 对指定的bean所有的方法(了解)

bean('userServiceId')

②、springAOP 的具体加载步骤:

  1、当 spring 容器启动的时候,加载了 spring 的配置文件

  2、为配置文件中的所有 bean 创建对象

  3、spring 容器会解析 aop:config 的配置

    1、解析切入点表达式,用切入点表达式和纳入 spring 容器中的 bean 做匹配

    如果匹配成功,则会为该 bean 创建代理对象,代理对象的方法=目标方法+通知

    如果匹配不成功,不会创建代理对象

  4、在客户端利用 context.getBean() 获取对象时,如果该对象有代理对象,则返回代理对象;如果没有,则返回目标对象

    说明:如果目标类没有实现接口,则 spring 容器会采用 cglib 的方式产生代理对象,如果实现了接口,则会采用 jdk 的方式

import org.aspectj.lang.JoinPoint;

import org.aspectj.lang.ProceedingJoinPoint;

public class StudentServiceAspect {

public void doBefore(JoinPoint jp){

System.out.println("类名:"+jp.getTarget().getClass().getName());

System.out.println("方法名:"+jp.getSignature().getName());

System.out.println("开始添加学生:"+jp.getArgs()[0]);

}

public void doAfter(JoinPoint jp){

System.out.println("类名:"+jp.getTarget().getClass().getName());

System.out.println("方法名:"+jp.getSignature().getName());

System.out.println("学生添加完成:"+jp.getArgs()[0]);

}

public Object doAround(ProceedingJoinPoint pjp) throws Throwable{

System.out.println("添加学生前");

Object retVal=pjp.proceed();

System.out.println(retVal);

System.out.println("添加学生后");

return retVal;

}

public void doAfterReturning(JoinPoint jp){

System.out.println("返回通知");

}

public void doAfterThrowing(JoinPoint jp,Throwable ex){

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

System.out.println("异常信息:"+ex.getMessage());

}

}

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">

public class StudentServiceImpl implements StudentService{

@Override

public void addStudent(String name) {

// System.out.println("开始添加学生"+name);

System.out.println("添加学生"+name);

System.out.println(1/0);

// System.out.println("完成学生"+name+"的添加");

}

}



AspectJ是一个面向切面的框架,它扩展了Java语言。

Aspect 通知类型,定义了类型名称以及方法格式。类型如下:

before:前置通知(应用:各种校验)

在方法执行前执行,如果通知抛出异常,阻止方法运行

afterReturning:后置通知(应用:常规数据处理)

方法正常返回后执行,如果方法中抛出异常,通知无法执行

必须在方法执行后才执行,所以可以获得方法的返回值。

around:环绕通知(应用:十分强大,可以做任何事情)

方法执行前后分别执行,可以阻止方法的执行

必须手动执行目标方法

afterThrowing:抛出异常通知(应用:包装异常信息)

方法抛出异常后执行,如果方法没有抛出异常,无法执行

after:最终通知(应用:清理现场)

方法执行完毕后执行,无论方法中是否出现异常

这里最重要的是around,环绕通知,它可以代替上面的任意通知。

在程序中表示的意思如下:

try{

//前置:before

//手动执行目标方法

//后置:afterRetruning

} catch(){

//抛出异常 afterThrowing

} finally{

//最终 after

}

AOP具体实例

public interface UserService {

//添加 user

public void addUser();

//删除 user

public void deleteUser();

}

创建实现类

public class UserServiceImpl implements UserService{

@Override

public void addUser() {

System.out.println("增加 User");

}

@Override

public void deleteUser() {

System.out.println("删除 User");

}

}

创建切面类(包含各种通知)

import org.aspectj.lang.JoinPoint;

public class MyAspect {

/**

* JoinPoint 能获取目标方法的一些基本信息

* @param joinPoint

*/

public void myBefore(JoinPoint joinPoint){

System.out.println("前置通知 : " + joinPoint.getSignature().getName());

}

public void myAfterReturning(JoinPoint joinPoint,Object ret){

System.out.println("后置通知 : " + joinPoint.getSignature().getName() + " , -->" + ret);

}

public void myAfter(){

System.out.println("最终通知");

}

}

创建spring配置文件applicationContext.xml

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">

@Test

public void testAop(){

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

测试异常通知

public

 void addUser() {

int i = 1/0;//显然这里会抛出除数不能为 0

System.out.println("增加 User");

}

切面类

public void myAfterThrowing(JoinPoint joinPoint,Throwable e){

System.out.println("抛出异常通知 : " + e.getMessage());

}

@Test

public void testAop(){

String str = "com/my/execption/applicationContext.xml";

ApplicationContext context = new ClassPathXmlApplicationContext(str);

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

测试环绕通知

public class MyAspect {

public Object myAround(ProceedingJoinPoint joinPoint) throws Throwable{

System.out.println("前置通知");

//手动执行目标方法

Object obj = joinPoint.proceed();

System.out.println("后置通知");

return obj;

}

}

applicationContext.xml 配置如下:

通过 xml 配置的方式

@Test

public void testAop(){

String str = "com/my/around/applicationContext.xml";

ApplicationContext context = new ClassPathXmlApplicationContext(str);

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

注解实现 AOP

 @Aspect 声明切面,修饰切面类,从而获得 通知。

  通知

    @Before 前置

    @AfterReturning 后置

    @Around 环绕

    @AfterThrowing 抛出异常

    @After 最终

  切入点

    @PointCut ,修饰方法 private void xxx(){} 之后通过“方法名”获得切入点引用

在 applicationContext.xml 文件中导入相应的命名空间

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:context="http://www.springframework.org/schema/context"

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

http://www.springframework.org/schema/context

http://www.springframework.org/schema/context/spring-context.xsd">

注解配置 bean

 xml配置:


配置扫描注解识别

在 applicationContext.xml 文件中添加如下配置:

在切面类上添加 @Aspect 注解,如下:

告诉 Spring 哪个是切面类


如何让 Spring 认识我们所配置的 AOP 注解呢

 我们在 applicationContext.xml 文件中增加如下配置:

注解配置前置通知

  我们先看 xml 配置前置通知如下:

那么注解的方式如下:


注解配置后置通知

  xml 配置后置通知:

注意看,后置通知有个 returning="ret" 配置,这是用来获得目标方法的返回值的。


public void testAopAnnotation(){

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext_Annotation.xml");

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

useService.deleteUser();

}

你可能感兴趣的:(Spring详解(四))