摘要

本文内容为我在网上搜集Spring AOP资料的汇总、摘抄。

AOP是一种编程思想,其对不同对象进行了横向的抽象,将不同对象的、和主流程无关的公共逻辑抽象出来以方便维护。AOP的实现基础为AOP动态代理,动态代理又可以由JDK动态代理和CGLIB实现。Spring中AOP的编程模型是定义组件、定义切入点、定义增强功能。

关键词:SpringAOP

一、AOP简介

1. 什么是AOP

AOP(Aspect Oriented Programming,面向切面编程),横向抽象的一种编程思想,是对OOP(Object Oriented Programming,面向对象编程)的补充和完善。

OOP中的基本三个基本特征:封装继承多态,其对象关系的描述是纵向的,对于分散在不同对象、同一层面的代码却无能为力。比如需要在代码开头和结尾处都打印相同的日志,就没有办法描述这种业务了。

AOP则可以解决这个问题,它可以将不同代码同一层次的业务无关的公共逻辑抽象出来,再把代码逻辑织入到对应的切面点,这样就可以减少重复代码,以及方便维护了。

AOP的应用可以有权限认证、日志、事务等。

2. AOP的一些概念

概念 解释
核心关注点 业务处理的主要流程为核心关注点
横切关注点 与业务处理主要流程不相关的部分为横切关注点(特征:分散
切面(aspect) 切面为对横切关注点的抽象
连接点(joinpoint) 被拦截的点,在Spring中为拦截到的方法
切入点(pointcut) 对连接点进行拦截的定义
通知(adivce) 指拦截到连接点后,要执行的代码。可以分为:前置后置异常最终环绕
目标对象 代理的目标对象
织入(weave) 将切面应用到目标对象,产生代理对象的过程。(即将行为织入到业务流程中)
引入(introduction) 不修改代码,在运行期(runtime)将类动态添加一些方法或字段。

二、Spring中AOP的实现原理

Spring AOP实现的基石在于代理,也就是应用AOP后实际上是生成AOP代理对象,Spring中AOP代理及其依赖关系由IoC容器负责生成、管理。

代理对象=目标对象+切面处理(通知)

1. 代理

1.1 静态代理

静态代理产生于代码的编译阶段,一旦代码运行就不可改变;

// 接口
public interface IPerson {
  public void doSth();
}

// 接口实现
public class Person implements IPerson {
  public void doSth(){
    System.out.println("do sth");
  }
}

// 代理实现
public class PersonProxy {
  private IPerson person;

  public PersonProxy(IPerson person) {
    this.person = person;
  }

  public void doSth() {
    System.out.println("before do sth");
    person.doSth();
    System.out.println("after do sth");
  }
}
public static void main(String[] args) {
  PersonProxy personProxy = new PersonProxy(new Person());
  personProxy.doSth();
}
/*
result console:
before do sth
do sth
after do sth
*/

1.2 动态代理

这里特指JDK动态代理,区别与静态代理,动态代理产生于代码的运行时阶段。JDK动态代理利用了Java的反射机制,生成一个实现代理接口的类,调用具体方法由InvocationHandler来处理。这种动态代理需要依赖接口来实现。

public class PersonProxy implements InvocationHandler {
  private Object delegate;

  public Object bind(Object delegate) {
    this.delegate = delegate;
    // 指明类加载器、指明生成哪个对象的代理对象(接口指定)、指明Handler
    return Proxy.newProxyInstance(delegate.getClass().getClassLoader(), delegate.getClass().getInterfaces(), this);
  }

  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    Object result = null;
    try {
      System.out.println("before do sth");
      result = method.invoke(delegate, args);
      System.out.println("after do sth");
    } catch(Exception e) {
      throw e;
    }
    return result;
  }
}
public static void main(String[] args) {
  PersonProxy personProxy = new PersonProxy();
  IPerson iPerson = (IPerson) personProxy.bind(new Person());
  iPerson.doSth();
}
/*
result console:
before do sth
do sth
after do sth
*/

1.3 CGLIB

如果没有接口,动态代理可以借助CGLIB来实现,它可以用生成代理目标子类(因此不能代理final修饰的类)的方式来实现代理,而不是接口,规避来接口的局限性。

实现MethodInterceptor接口来实现。

public class PersonProxy implements MethodInterceptor {
  private Object delegate;
  public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
    System.out.println("before do sth");
    Object result = methodProxy.invokeSuper(method, args);
    System.out.println("after do sth");
    return result;
  }
  public static Person getProxyInstance() {
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(Person.class);
    enhancer.setCallback(new PersonProxy());
    return (Person) enhancer.create();
  }
}
public static void main(String[] args) {
  Person person = PersonProxy.getProxyInstance();
  person.doSth();
}
/*
result console:
before do sth
do sth
after do sth
*/

2. Spring AOP代理创建规则

Spring提供了两种方式来生成代理对象:JDKProxyCGLIB
使用方式由AopProxyFactory根据AdvisedSupport对象配置来决定。

代理默认创建规则:

  1. 如果目标类是接口,则使用Java动态代理创建AOP代理;
  2. 如果目标类不是接口,则使用Cglib生成代理;

三、在Spring中使用AOP

AOP编程步骤:

  1. 定义普通业务组件;
  2. 定义切入点(切入点和业务组件的对应关系可以是一对多);
  3. 定义增强处理,即为普通业务组件织入的动作;

1. 使用XML文件配置AOP

参考:Spring 中基于 AOP 的 XML架构

2. 使用注解配置AOP

参考:Spring 中基于 AOP 的 @AspectJ

四、总结

  1. AOP(Aspect Oriented Programming,面向切面编程),横向抽象的一种编程思想,是对OOP(Object Oriented Programming,面向对象编程)的补充和完善。
  2. Spring AOP实现的基石在于代理,也就是应用AOP后实际上是生成AOP代理对象,Spring中AOP代理及其依赖关系由IoC容器负责生成、管理。
  3. Spring中AOP的编程模型是定义组件、定义切入点、定义增强功能。

参考资料

  1. Spring3:AOP
  2. Spring AOP原理分析一次看懂
  3. Spring-aop 全面解析(从应用到原理)
  4. Spring的两种代理JDK和CGLIB的区别浅谈
  5. Spring 教程

Spring:AOP的更多相关文章

  1. Spring:AOP面向切面编程

    AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果. AOP是软件开发思想阶段性的产物,我们比较熟悉面向过程O ...

  2. spring:AOP面向切面编程(注解)03

    使用注解写aop时最好使用环绕通知写 切面类: /** * 用于记录日志的工具类,它里面提供了公共的代码 */ @Component("logger") @Aspect //表示当 ...

  3. spring:AOP面向切面编程02

    参考: https://blog.csdn.net/jeffleo/article/details/54136904 一.AOP的核心概念AOP(Aspect Oriented Programming ...

  4. 理解Spring:AOP的原理及手动实现

    引入 到目前为止,我们已经完成了简易的IOC和DI的功能,虽然相比如Spring来说肯定是非常简陋的,但是毕竟我们是为了理解原理的,也没必要一定要做一个和Spring一样的东西.到了现在并不能让我们松 ...

  5. Spring:Aop before after afterReturn afterThrowing around 的原理

    在写这篇文章前,在网上看了一下大多数的文章,在说这一块时,都是用语言来表达的.before.after.afterReturn.afterThrowing 这四个用语言是可以说清楚的,但 around ...

  6. 使用spring:aop中修改增强方法中的参数

    大家都知道,在增强方法中,使用jp.getArgs()[index]可以获取传进来的参数,但是参数传进来之后,怎么改变它的值呢? 因为jp.getArgs()[index]获取到的只是数据的备份,所以 ...

  7. 深入浅出学习Spring框架(三):AOP 详解

    AOP的英文解释——AOPAspect Oriented Programming面向切面编程.主要目的是通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术. 在反 ...

  8. Spring使用 --- 基本概念(二):AOP,面向方面编程

    Table of Contents 什么是面向方面编程 怎样使用 什么时候使用 好处 本文讲述sprint的第二个基本概念: AOP,即面向方面编程 什么是面向方面编程 软件项目中,日志系统等服务系统 ...

  9. Java框架spring 学习笔记(十二):aop实例操作

    使用aop需要在网上下载两个jar包: aopalliance.jar aspectjweaver.jar 为idea添加jar包,快捷键ctrl+shift+alt+s,打开添加jar包的对话框,将 ...

随机推荐

  1. 总结,为什么要重写hashset的hashcode()和equals()?

    看了非常多博客,怕自己忘记了,通俗易懂的总结如下 本人总结下: 重写前,比较地址,hashcode方法如果相等可能是同一个对象,所以再用equals再比内存地址 重写后,比较值,重写hashCode方 ...

  2. PAT 乙级 1074 宇宙无敌加法器 (20 分)

    1074 宇宙无敌加法器 (20 分) 地球人习惯使用十进制数,并且默认一个数字的每一位都是十进制的.而在 PAT 星人开挂的世界里,每个数字的每一位都是不同进制的,这种神奇的数字称为“PAT数”.每 ...

  3. 二叉树遍历(flist)(二叉树,已知中序层序,求先序)

    问题 C: 二叉树遍历(flist) 时间限制: 1 Sec  内存限制: 128 MB提交: 76  解决: 53[提交][状态][讨论版][命题人:quanxing][Edit] [TestDat ...

  4. python科学计算包numpy用法(一)

    numpy是python中一个用来做科学计算的包,用起来十分方便,下面是我总结的numpy的用法: 1.如何创建矩阵 创建矩阵有很多种方法,主要包括以下几种: 通过array函数创建 >> ...

  5. Egret飞行模拟-开发记录01

    1.项目结构简介 1.1 index.html:应用入口文件,我们可以在这里面配置项目的旋转缩放模式背景颜色等. 1.2 egretProperties.json:这个文件里面进行项目配置,包括模块和 ...

  6. 使用阿里云Java SDK 实现 DDNS

    本代码的实现前提: 1.拥有阿里云域名,且获取了Access Key 及 Access Secret 2.能获取外网IP的页面地址(注意:ip138.com的实际包含ip地址为http://2018. ...

  7. 4-HBase

    定义: HBase是一个分布式的.面向列的开源数据库,该技术来源于 Fay Chang 所撰写的Google论文“Bigtable:一个结构化数据的分布式存储系统”.就像Bigtable利用了Goog ...

  8. TreeMap中文排序,TreeMap倒序输出排列

    1.TreeMap集合倒序排列 import java.util.Comparator; /** * 比较算法的类,比较器 * @author Administrator * */ public cl ...

  9. Linux高级指令

    一.hostname指令 作用:操作服务器的主机名(读取,设置) #hostname    作用:表示输出完整的主机名 #hostname -f    作用:表示输出当前主机名中的FQDN(权限定域名 ...

  10. [python,2019-02-15] 最短回文串

    给定一个字符串 s,你可以通过在字符串前面添加字符将其转换为回文串.找到并返回可以用这种方式转换的最短回文串. 示例 1: 输入: "aacecaaa" 输出: "aaa ...