Spring框架:Spring AOP

本文内容纲要:

- 为什么需要AOP?

- 使用动态代理解决问题

- Spring AOP可以解决此类问题!

- AOP简介

- AOP术语

- Spring AOP

- 在 Spring 中启用 AspectJ 注解支持

- 用 AspectJ 注解声明切面

- 举个栗子

- 步骤

为什么需要AOP?

•代码混乱:越来越多的非业务需求(日志和验证等)加入后, 原有的业务方法急剧膨胀**.** 每个****方法在处理核心逻辑的同时还必须兼顾其他多个关注点.

•代码分散: 以日志需求为例, 只是为了满足这个单一需求, 就不得不在多个模块(方法)里多次重复相同的日志代码. 如果日志需求发生变化**,** 必须修改所有模块.

问题:

Image

普通代码:

ArithmeticCalculator接口

package com.yorkmass.spring.aop.helloworld;

public interface ArithmeticCalculator {

int add(int i,int j);

int sub(int i,int j);

int mul(int i,int j);

int div(int i,int j);

}

ArithmeticCalculatorLoggingImpl类

package com.yorkmass.spring.aop.helloworld;

public class ArithmeticCalculatorLoggingImpl implements ArithmeticCalculator {

@Override

public int add(int i, int j) {

System.out.println("The method add begins with["+i+","+j+"]");

int result=i+j;

System.out.println("The method add ends with"+result);

return result;

}

@Override

public int sub(int i, int j) {

System.out.println("The method sub begins with["+i+","+j+"]");

int result=i-j;

System.out.println("The method sub ends with"+result);

return result;

}

@Override

public int mul(int i, int j) {

System.out.println("The method mul begins with["+i+","+j+"]");

int result=i*j;

System.out.println("The method mul ends with"+result);

return result;

}

@Override

public int div(int i, int j) {

System.out.println("The method div begins with["+i+","+j+"]");

int result=i/j;

System.out.println("The method div ends with"+result);

return result;

}

}

然后自己写测试类运行即可

但是上面如果后期有需求需要对代码进行修改,则需要对每一行的日志文件进行修改,特别不方便修改。

我们能不能去掉每个方法的输入语句,只写一个,来实现对所有方法的日志管理,这样修改起来也比较方便,

动态代理便可以很好的解决这个问题。

使用动态代理解决问题

•代理设计模式的原理: 使用一个代理将对象包装起来, 然后用该代理对象取代原始对象. 任何对原始对象的调用都要通过代理. 代理对象决定是否以及何时将方法调用转到原始对象上.

Image

动态代理解决办法:

我们把 ArithmeticCalculatorImpl类写的尽可能简单

ArithmeticCalculatorImpl类

package com.yorkmass.spring.aop.helloworld;

public class ArithmeticCalculatorImpl implements ArithmeticCalculator {

@Override

public int add(int i, int j) {

int result=i+j;

return result;

}

@Override

public int sub(int i, int j) {

int result=i-j;

return result;

}

@Override

public int mul(int i, int j) {

int result=i*j;

return result;

}

@Override

public int div(int i, int j) {

int result=i/j;

return result;

}

}

我们开始配置动态代理

ArithmeticCalculatorLoggingProxy类

package com.yorkmass.spring.aop.helloworld;

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;

import java.util.Arrays;

import javax.naming.spi.DirStateFactory.Result;

import org.omg.CORBA.PUBLIC_MEMBER;

public class ArithmeticCalculatorLoggingProxy {

//要代理的对象

private ArithmeticCalculator targer;

public ArithmeticCalculatorLoggingProxy(ArithmeticCalculator target) {

// TODO Auto-generated constructor stub

this.targer=target;

}

public ArithmeticCalculator getLoggingProxy(){

ArithmeticCalculator proxy=null;

//代理对象由哪一个类加载器负责加载

ClassLoader loader=targer.getClass().getClassLoader();

//代理对象的类型,即其中有哪些方法

Class[] interfaces=new Class[]{ArithmeticCalculator.class};

//当调用代理对象其中的方法时,该执行的代码

InvocationHandler h=new InvocationHandler() {

/**

* proxy:正在返回的那个代理对象,一般情况下,在invoke方法中都不适用该对象。

* method:正在调用的方法

* args:调用方法时,传入的参数

*/

@Override

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

String methodName=method.getName();

//日志

System.out.println("The method..."+methodName+"begins with"+Arrays.asList(args));

//执行方法

Object result=method.invoke(targer, args);

//日志

System.out.println("The method"+methodName+"ends with "+result);

return result;

}

};

proxy=(ArithmeticCalculator)Proxy.newProxyInstance(loader, interfaces, h);

return proxy;

}

}

测试主类Main:

package com.yorkmass.spring.aop.helloworld;

public class Main {

public static void main(String[] args) {

// TODO Auto-generated method stub

// ArithmeticCalculator arithmeticCalculator=null;

// arithmeticCalculator=new ArithmeticCalculatorLoggingImpl();

ArithmeticCalculator target=new ArithmeticCalculatorImpl();

ArithmeticCalculator proxy=new ArithmeticCalculatorLoggingProxy(target).getLoggingProxy();

int result=proxy.add(1, 2);

System.out.println("-->"+result);

result=proxy.div(4, 2);

System.out.println("-->"+result);

}

}

运行结果:

The method...addbegins with[1, 2]

The methodaddends with 3

-->3

The method...divbegins with[4, 2]

The methoddivends with 2

-->2

Spring AOP可以解决此类问题!

AOP简介

•AOP(Aspect-Oriented Programming, 面向切面编程): 是一种新的方法论, 是对传统 OOP(Object-Oriented Programming, 面向对象编程) 的补充.

•AOP 的主要编程对象是切面(aspect), 而切面模块化横切关注点.

•在应用 AOP 编程时, 仍然需要定义****公共功能, 但可以明确的定义这个功能在哪里, 以什么方式应用, 并且不必修改受影响的类. 这样一来**横切关注点就被模块化到特殊的对象****(切面)**里.

•AOP 的好处:

–每个事物逻辑位于一个位置, 代码不分散, 便于维护和升级

–业务模块更简洁, 只包含核心业务代码.

Image

AOP术语

•切面(Aspect): 横切关注点**(跨越应用程序多个模块的功能)**被模块化的特殊对象

•通知(Advice): 切面必须要完成的工作

•目标(Target): 被通知的对象

•代理(Proxy): 向目标对象应用通知之后创建的对象

•连接点(Joinpoint):程序****执行的某个特定位置:如类某个方法调用前、调用后、方法抛出异常后等。连接点****由两个信息确定:方法表示的程序执行点;相对点表示的方位。例如 ArithmethicCalculator#add() 方法执行前的连接点,执行点为 ArithmethicCalculator#add(); 方位为该方法执行前的位置

•切点(pointcut):每个****类都拥有多个连接点:例如 ArithmethicCalculator 的所有方法实际上都是连接点,即连接点是程序类中客观存在的事务。AOP 通过切点定位到特定的连接点**。类比****:连接点相当于数据库中的记录,切点相当于查询条件**。切点和连接点不是一对一的关系,一个切点匹配多个连接点,切点通过 org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件。

Spring AOP

•AspectJ:Java 社区里最完整最流行的 AOP 框架.

•在 Spring2.0 以上版本中, 可以使用基于 AspectJ 注解或基于 XML 配置的 AOP

在 Spring 中启用 AspectJ 注解支持

•要在 Spring 应用中使用 AspectJ 注解, 必须在 classpath 下包含 AspectJ 类库: aopalliance.jar、aspectj.weaver.jar 和 spring-aspects.jar

•将 aop Schema 添加到 根元素中.

•要在 Spring IOC 容器中启用 AspectJ 注解支持, 只要在 Bean 配置文件中定义一个空的 XML 元素 <aop:aspectj-autoproxy>

•当 Spring IOC 容器侦测到 Bean 配置文件中的 aop:aspectj-autoproxy 元素时, 会自动为与 AspectJ 切面匹配的 Bean 创建代理.

用 AspectJ 注解声明切面

•要在 Spring 中声明 AspectJ 切面**,** 只需要在 IOC 容器中将切面声明为 Bean 实例. 当在 Spring IOC 容器中初始化 AspectJ 切面之后, Spring IOC 容器就会为那些与 AspectJ 切面相匹配的 Bean 创建代理.

•在 AspectJ 注解中**,** 切面只是一个带有 @Aspect 注解的 Java 类.

•通知是标注有某种注解的简单的 Java 方法.

•AspectJ 支持 5 种类型的通知注解:

–@Before: 前置通知, 在方法执行之前执行

–@After: 后置通知, 在方法执行之后执行

–@AfterRunning: 返回通知, 在方法返回结果之后执行

–@AfterThrowing: 异常通知, 在方法抛出异常之后

–@Around: 环绕通知, 围绕着方法执行

举个栗子

前置、返回、后置、异常、环绕通知

前置通知

•前置通知:在方法执行之前执行的通知

•前置通知使用 @Before 注解, 并将切入点表达式的值作为注解值.

后置通知同理

Image

利用方法签名编写 AspectJ 切入点表达式

•最典型的切入点表达式时根据方法的签名来匹配各种方法:

–execution * com.atguigu.spring.ArithmeticCalculator.*(..): 匹配 ArithmeticCalculator 中声明的所有方法,第一个 * 代表任意修饰符及任意返回值. 第二个 * 代表任意方法. .. 匹配任意数量的参数. 若目标类与接口与该切面在同一个包中, 可以省略包名.

–execution public * ArithmeticCalculator.*(..): 匹配 ArithmeticCalculator 接口的所有公有方法.

–execution public double ArithmeticCalculator.*(..): 匹配 ArithmeticCalculator 中返回 double 类型数值的方法

–execution public double ArithmeticCalculator.*(double, ..): 匹配第一个参数为 double 类型的方法, .. 匹配任意数量任意类型的参数

–execution public double ArithmeticCalculator.*(double, double): 匹配参数类型为 double, double 类型的方法.

合并切入点表达式

•在 AspectJ 中, 切入点表达式可以通过操作符 &&, ||, ! 结合起来.

Image

让通知访问当前连接点的细节

Image

步骤

1).新建bin文件夹,导入 jar 包:

com.springsource.org.aopalliance-1.0.0.jar

com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

spring-aop-4.0.0.RELEASE.jar

spring-aspects-4.0.0.RELEASE.jar

commons-logging-1.1.1.jar

spring-beans-4.0.0.RELEASE.jar

spring-context-4.0.0.RELEASE.jar

spring-core-4.0.0.RELEASE.jar

spring-expression-4.0.0.RELEASE.jar

Image

2).新建配置文件如:applicationContext.xml,在配置文件中加入aop的命名空间

Image

3)

①.基于注解的方式

<context:component-scan base-package="com.yorkmass.spring.aop.impl"></context:component-scan>

②.在配置文件中加入如下配置

aop:aspectj-autoproxy</aop:aspectj-autoproxy>

③.吧横切关注点的代码抽象到切面的类中

i.切面首先是一个IOC中的bean,即加入@Component注释

ii.切面还需要加入@Aspect注解

④.在类中声明各种通知:

i:声明一个方法

ii:在方法前加入@Before注解

execution(public int com.yorkmass.spring.aop.impl.ArithmeticCalculator.*(int, int))

execution:执行的意思

也可以改为* com.yorkmass.spring.aop.impl.*.*(int, int)

上面表示为任意修饰符任意返回值impl包里面的所有类所有方法

⑤.可以在通知方法中声明一个类型为 JoinPoint 的参数. 然后就能访问链接细节. 如方法名称和参数值.

@Aspect

@Component

public class LoggingAspect {

    //声明该方法是一个前置通知:在目标方法开始之前执行

    @Before("execution(public int com.yorkmass.spring.aop.impl.ArithmeticCalculator.*(int, int))")

    public void beforeMethod(JoinPoint joinPoint){

        String methodName=joinPoint.getSignature().getName();

        List<Object> args=Arrays.asList(joinPoint.getArgs());

        System.out.println("The method "+methodName+" begins with"+args);

    }  

}

新建一个包com.yorkmass.spring.aop.impl

下面新建3个类:ArithmeticCalculator、ArithmeticCalculatorImpl、LoggingAspect

新建一个测试类Main

程序结构:

Image

ArithmeticCalculator接口

package com.yorkmass.spring.aop.impl;

public interface ArithmeticCalculator {

int add(int i,int j);

int sub(int i,int j);

int mul(int i,int j);

int div(int i,int j);

}

ArithmeticCalculatorImpl类

package com.yorkmass.spring.aop.impl;

import org.springframework.stereotype.Component;

@Component

public class ArithmeticCalculatorImpl implements ArithmeticCalculator {

@Override

public int add(int i, int j) {

int result=i+j;

return result;

}

@Override

public int sub(int i, int j) {

int result=i-j;

return result;

}

@Override

public int mul(int i, int j) {

int result=i*j;

return result;

}

@Override

public int div(int i, int j) {

int result=i/j;

return result;

}

}

LoggingAspect类

package com.yorkmass.spring.aop.impl;

import java.util.Arrays;

import java.util.List;

import org.aspectj.lang.JoinPoint;

import org.aspectj.lang.annotation.Aspect;

import org.aspectj.lang.annotation.Before;

import org.springframework.stereotype.Component;

//把这个类声明为一个切面:需要把该类放入到IOC容器中,再声明为一个切面

@Aspect

@Component

public class LoggingAspect {

//声明该方法是一个前置通知:在目标方法开始之前执行

@Before("execution(public int com.yorkmass.spring.aop.impl.ArithmeticCalculator.*(int, int))")

public void beforeMethod(JoinPoint joinPoint){

String methodName=joinPoint.getSignature().getName();

List<Object> args=Arrays.asList(joinPoint.getArgs());

System.out.println("The method "+methodName+" begins with"+args);

}

//后置通知:在目标方法执行后(无论是否发生异常),执行的通知

//在后置通知中还不能访问目标方法执行的结果

@After("execution(* com.yorkmass.spring.aop.impl.*.*(int,int))")

public void afterMethod(JoinPoint joinPoint){

String methodName=joinPoint.getSignature().getName();

System.out.println("The method "+methodName+"ends");

}

/**

* 在方法正常结束后执行的代码

* 返回通知是可以访问到方法的返回值的!

* @param joinPoint

*/

@AfterReturning(value="execution(public int com.yorkmass.spring.aop.ArithmeticCalculator.*(..))",

returning="result")

public void afterReturning(JoinPoint joinPoint,Object result){

String methodName=joinPoint.getSignature().getName();

System.out.println("The method "+methodName+" ends with "+result);

}

/**

* 在方法出现异常时会执行的代码

* 可以访问到异常对象,且可以指定在出现特定异常时再执行通知代码

* @param joinPoint

* @param ex

*/

@AfterThrowing(value="execution(public int com.yorkmass.spring.aop.ArithmeticCalculator.*(..))",

throwing="ex")

// public void afterThrowing(JoinPoint joinPoint,Exception ex){

// String methodName=joinPoint.getSignature().getName();

// System.out.println("The method "+methodName+" Throwing:"+ex);

// }

public void afterThrowing(JoinPoint joinPoint,NullPointerException ex){

String methodName=joinPoint.getSignature().getName();

System.out.println("The method "+methodName+" Throwing:"+ex);

}

/**

* 环绕通知需要携带ProceedingJoinPoint类型的参数

* 环绕通知相当于动态代理的全过程,ProceedingJoinPoint类型的参数可以决定是否执行目标方法。

* 且环绕通知必须有返回值,返回值即为目标方法的返回值

*/

@Around("execution(public int com.yorkmass.spring.aop.ArithmeticCalculator.*(..))")

public Object aroundMethod(ProceedingJoinPoint pjd){

Object result=null;

String methodName=pjd.getSignature().getName();

//执行目标方法

try {

//前置通知

System.out.println("The method "+methodName+" begin with: "+Arrays.asList(pjd.getArgs()));

result=pjd.proceed();

//返回通知

System.out.println("The method "+methodName+" end with:"+result);

} catch (Throwable e) {

// TODO: handle exception

//异常通知

System.out.println("The method "+methodName+" occurs exception:"+e);

throw new RuntimeException(e);

}

//后置通知

System.out.println("The method "+methodName+" end with");

return result;

}

}

applicationContext.xml

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

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

xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd

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

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

<!-- 配置自动扫描的包 -->

<context:component-scan base-package="com.yorkmass.spring.aop.impl"></context:component-scan>

<!-- 使AspjectJ 注解起作用 :自动为匹配的类生成代理对象-->

<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

</beans>

Main类

package com.yorkmass.spring.aop;

import org.springframework.context.ApplicationContext;

import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Main {

public static void main(String[] args) {

// TODO Auto-generated method stub

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

ArithmeticCalculator arithmeticCalculator=(ArithmeticCalculator)ctx.getBean("arithmeticCalculator");

int result=arithmeticCalculator.add(1, 5);

System.out.println("result:"+result);

result=arithmeticCalculator.div(1000, 0);

System.out.println("result:"+result);

}

}

运行结果:

一月 19, 2019 1:39:50 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh

信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@1c2faae: startup date [Sat Jan 19 13:39:50 CST 2019]; root of context hierarchy

一月 19, 2019 1:39:50 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions

信息: Loading XML bean definitions from class path resource [applicationContext.xml]

The method add begin with: [1, 5]

The method add begins with [1, 5]

The method add end with:6

The method add end with

The method add ends with

The method add ends with 6

result:6

The method div begin with: [1000, 0]

The method div begins with [1000, 0]

The method div occurs exception:java.lang.ArithmeticException: / by zero

The method div ends with

Exception in thread "main" java.lang.RuntimeException: java.lang.ArithmeticException: / by zero

at com.yorkmass.spring.aop.LoggingAspect.aroundMethod(LoggingAspect.java:86)

at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)

at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)

at java.lang.reflect.Method.invoke(Unknown Source)

at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethodWithGivenArgs(AbstractAspectJAdvice.java:621)

at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethod(AbstractAspectJAdvice.java:610)

at org.springframework.aop.aspectj.AspectJAroundAdvice.invoke(AspectJAroundAdvice.java:68)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.aspectj.AspectJAfterAdvice.invoke(AspectJAfterAdvice.java:43)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.framework.adapter.AfterReturningAdviceInterceptor.invoke(AfterReturningAdviceInterceptor.java:52)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.aspectj.AspectJAfterThrowingAdvice.invoke(AspectJAfterThrowingAdvice.java:58)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:92)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:207)

at com.sun.proxy.$Proxy11.div(Unknown Source)

at com.yorkmass.spring.aop.Main.main(Main.java:15)

Caused by: java.lang.ArithmeticException: / by zero

at com.yorkmass.spring.aop.ArithmeticCalculatorImpl.div(ArithmeticCalculatorImpl.java:31)

at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)

at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)

at java.lang.reflect.Method.invoke(Unknown Source)

at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:317)

at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:190)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:157)

at org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor.invoke(MethodBeforeAdviceInterceptor.java:52)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)

at org.springframework.aop.aspectj.MethodInvocationProceedingJoinPoint.proceed(MethodInvocationProceedingJoinPoint.java:85)

at com.yorkmass.spring.aop.LoggingAspect.aroundMethod(LoggingAspect.java:78)

... 19 more

本文内容总结:为什么需要AOP?,使用动态代理解决问题,Spring AOP可以解决此类问题!,AOP简介,AOP术语,Spring AOP,在 Spring 中启用 AspectJ 注解支持,用 AspectJ 注解声明切面,举个栗子,步骤,

原文链接:https://www.cnblogs.com/yorkmass/p/11109874.html

以上是 Spring框架:Spring AOP 的全部内容, 来源链接: utcz.com/z/296668.html

回到顶部