spring aop学习

2 篇文章 0 订阅

  AOP(Aspect Oriented Programming),即面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待等。

  AOP可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善。OOP引入封装、继承、多态等概念来建立一种对象层次结构,用于模拟公共行为的一个集合。不过OOP允许开发者定义纵向的关系,但并不适合定义横向的关系,例如日志功能。日志代码往往横向地散布在所有对象层次中,而与它对应的对象的核心功能毫无关系对于其他类型的代码,如安全性、异常处理和透明的持续性也都是如此,这种散布在各处的无关的代码被称为横切(cross cutting),在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

  AOP技术恰恰相反,它利用一种称为”横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为”Aspect”,即切面。所谓”切面”,简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。

一、 AOP的基本概念

(1) JointPoint(连接点):程序执行过程中某个特定的位置,如类初始化前后、方法调用前后、抛出异常时,连接点是程序中客观存在的事物

(2) Pointcut(切点):就是带有增强的连接点,在程序中主要体现为书写切入点表达式。连接点相当于数据库中的记录,切点相当于查询条件,切点与连接点不是一一对应的关系,一个切点可以对应多个连接点

(3) Aspect(切面):通常是一个类,里面可以定义切入点和增强代码。AOP的工作重心在于两点:1. 如何通过切点和增强定位到连接点上;2. 如何在增强中编写切面代码。

(4) Advice(增强):AOP在特定的切入点上织入的一段程序代码,有before,after,afterReturning,afterThrowing,around

(5) AOP代理(proxy):AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类

(6) 目标对象(Target):增强逻辑的织入目标类。如果没有AOP,那么目标业务类需要自己实现所有的逻辑

(7) 织入(Weaving):织入是将增强添加到目标类的具体连接点上的过程

(8) 引介(introduction):在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段

二、 Spring对AOP的支持

Spring中AOP代理由Spring的IOC容器负责生成、管理,其依赖关系也由IOC容器负责管理。因此,AOP代理可以直接使用容器中的其它bean实例作为目标,这种关系可由IOC容器的依赖注入提供。Spring创建代理的规则为:

1、默认使用Java动态代理来创建AOP代理,这样就可以为任何接口实例创建代理了

2、当需要代理的类不是代理接口的时候,Spring会切换为使用CGLIB代理,也可强制使用CGLIB

AOP编程其实是很简单的事情,纵观AOP编程,程序员只需要参与三个部分:

1、定义普通业务组件

2、定义切入点,一个切入点可能横切多个业务组件

3、定义增强处理,增强处理就是在AOP框架为普通业务组件织入的处理动作

所以进行AOP编程的关键就是定义切入点和定义增强处理,一旦定义了合适的切入点和增强处理,AOP框架将自动生成AOP代理,即:代理对象的方法=增强处理+被代理对象的方法。

三、 实例
建立一个maven搭建的spring web项目,这里主要使用了xml的配置方式,新建一个bean.xml的配置文件,基础配置如下:

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:p="http://www.springframework.org/schema/p"
       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">

pom.xml配置如下:

<dependencies>
    <!--dependency>
      <groupId>chapter7</groupId>
      <artifactId>[the artifact id of the block to be mounted]</artifactId>
      <version>1.0-SNAPSHOT</version>
    </dependency-->
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-core</artifactId>
        <version>5.0.3.RELEASE</version>
      </dependency>
      <!-- https://mvnrepository.com/artifact/org.springframework/spring-beans -->
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-beans</artifactId>
        <version>5.0.3.RELEASE</version>
      </dependency>
      <!-- https://mvnrepository.com/artifact/org.springframework/spring-test 测试-->


    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-aop</artifactId>
      <version>5.0.3.RELEASE</version>
    </dependency>

    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context-support</artifactId>
      <version>5.0.3.RELEASE</version>
    </dependency>
    <dependency>
      <groupId>org.testng</groupId>
      <artifactId>testng</artifactId>
      <version>6.1.1</version>
      <scope>test</scope>
      <exclusions>
        <exclusion>
          <artifactId>junit</artifactId>
          <groupId>junit</groupId>
        </exclusion>
      </exclusions>
    </dependency>
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>RELEASE</version>
    </dependency>

    <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.8.13</version>
    </dependency>
  </dependencies>

织入的目标对象:
接口

package com;

public interface Waiter {
    void greetTo(String name);
    void serveTo(String name);
}

实现类

package com;

public class NaiveWaiter implements Waiter {
    public void greetTo(String name){
        System.out.println("greet to "+name+"...");
    }

    public void serveTo(String name){
        System.out.println("serving "+name+"...");
    }
}

增强代码(包含前置增强、后置增强、环绕增强)

package com;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;

import java.text.SimpleDateFormat;
import java.util.Date;

public class aopAdvice {
    private SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SSSS");

    public void before(JoinPoint joinPoint){
        Object[] name = joinPoint.getArgs();
        System.out.println("How are you! Mr."+name[0]);
    }

    public void after(){
        System.out.println("Good bye!");
    }

    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        Object[] name = proceedingJoinPoint.getArgs();
        System.out.println("Good day! Mr."+name[0]+df.format(new Date()));
        proceedingJoinPoint.proceed();
        System.out.println("See you next time!"+df.format(new Date()));
    }
}

aop配置(xml)

    <bean id="target" class="com.NaiveWaiter"/>
    <bean id="advice" class="com.aopAdvice"/>

    <aop:config>
        <aop:aspect ref="advice">
            <aop:pointcut id="p" expression="execution(* com.NaiveWaiter..*(..))"></aop:pointcut>
            <aop:before pointcut-ref="p" method="before"/>
            <aop:after-returning pointcut-ref="p" method="after"/>
            <aop:around pointcut-ref="p" method="around"/>
        </aop:aspect>
    </aop:config>

测试类(使用代理工厂和AspectJ实现)

package com;

import org.junit.Test;
import org.springframework.aop.AfterAdvice;
import org.springframework.aop.BeforeAdvice;
import org.springframework.aop.framework.ProxyFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class BeforeAdviceTest {
    @Test
    public void before(){
        Waiter target = new NaiveWaiter();
        BeforeAdvice advice1 = new GreetingBeforeAdvice();
        AfterAdvice advice2 = new GreetingAfterAdvice();

        ProxyFactory pf = new ProxyFactory();
        pf.setInterfaces(target.getClass().getInterfaces());
        pf.setOptimize(true);

        pf.setTarget(target);

        pf.addAdvice(advice1);
        pf.addAdvice(advice2);

        Waiter proxy = (Waiter)pf.getProxy();
        proxy.greetTo("John");
        proxy.serveTo("Tom");
    }

    @Test
    public void BeforeAfter(){
        String config = "classpath:beans.xml";
        ApplicationContext context = new ClassPathXmlApplicationContext(config);
        Waiter waiter = (Waiter)context.getBean("target");
        waiter.greetTo("John");
        waiter.serveTo("Bob");
    }
}

结果:
代理工厂
这里写图片描述

AspectJ
这里写图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值