Spring 面向切面编程(AOP) D5

Spring AOP简介

问题提出

首先我们回顾一下OOP(Object Oriented Programming-面向对象编程),OOP引入了封装、继承、多态等概念建立了一种对象层次结构,用于模拟公共行为的一个集合。不过OOP允许开发者定义纵向的关系,并不适合定义横向的关系(例如日志功能)。 日志代码常常是横向的散布在所有对象层次中,这种散布在各处的重复的代码被称为横切(cross cutting)。如果仍然使用OOP设计,会导致大量的代码重复,不利于各模块的重用。因而我们引入AOP的编程思想。

面向切面编程

AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP的补充和完善。

在面向切面编程的思想里面,把功能分为核心业务功能和周边功能。

  • 核心业务: 比如登陆、CRUD等;
  • 周边功能: 比如性能统计、日志、事务管理等

上述的周边功能在Spring的AOP思想中,被定义为切面

在AOP思想里,核心业务功能和切面功能分别独立开发,随后把切面功能和核心业务“编织”在一起,这就叫AOP。

AOP 好处

AOP能够将那些和业务无关,却被业务模块共同调用的逻辑或责任(例如事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并且有利于未来的扩展和维护。

AOP 中的概念(理解即可)

  • 切入点(Pointcut)

    在哪些类、哪些方法上切入

  • 通知(Advice)

    在方法执行的什么时机(方法前?后?前和后?)增强什么功能

  • 切面(Aspect)

    切面 = 切入点 + 通知,即在什么时机,哪个地方,增强什么功能!

  • 织入(Weaving)

    把切面加入到对象中,并创建出代理对象的过程。(Spring实现)

AOP的一个案例

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KyR9ff08-1605142317545)(Spring%20%E9%9D%A2%E5%90%91%E5%88%87%E9%9D%A2%E7%BC%96%E7%A8%8B%EF%BC%88AOP%EF%BC%89%20D5.assets/image-20201110194758671.png)]

也可参照我的前一篇博客: Spring 代理模式

AOP 实现

问题背景说明:现有UserService接口(提供用户的CRUD),UserServiceImpl类(实现接口中方法)。如下:(均在service包下)

package service;

public interface UserService {
    public void add();
    public void delete();
    public void update();
    public void select();
}
package service;

public class UserServiceImpl implements UserService {
    public void add() {
        System.out.println("增加了一个用户");
    }

    public void delete() {
        System.out.println("删除了一个用户");
    }

    public void update() {
        System.out.println("更新信息");
    }

    public void select() {
        System.out.println("查询用户");
    }
}

需求: 对方法的执行实现追踪,即添加简易的日志功能!

注意: 使用aop织入,需要引入相应的依赖,代码如下:

<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.9.4</version>
</dependency>

方法一(使用原生Spring API接口)

首先创建log包,创建两个类:Log 、 AfterLog,分别实现相应的接口。代码如下:

Log:

package log;

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;

public class Log implements MethodBeforeAdvice {
    //method:要执行的目标对象的方法
    //objects: 参数
    //target: 目标对象
    public void before(Method method, Object[] objects, Object target) throws Throwable {
        System.out.println(o.getClass().getName()+"的"+method.getName()+"被执行了!");
    }
}

AfterLog:

package log;

import org.springframework.aop.AfterReturningAdvice;

import java.lang.reflect.Method;

public class AfterLog implements AfterReturningAdvice {
    public void afterReturning(Object returnValue, Method method, Object[] objects, Object target) throws Throwable {
        System.out.println("执行了"+method.getName()+",返回结果为:"+returnValue);
    }
}

applicationContext.xml中配置:(放在resources包下)

<?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/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--注册bean-->
    <bean id="userService" class="com.demut.service.UserServiceImpl"/>
    <bean id="log" class="com.demut.log.Log"/>
    <bean id="afterLog" class="com.demut.log.AfterLog"/>


    <!--方式一: 使用原生Spring API接口-->
    <!--配置aop: 需要导入aop的约束-->
    <aop:config>
        <!--切入点 : expression: 表达式,execution(要执行的位置!* * * * *)-->
        <aop:pointcut id="pointcut" expression="execution(* com.demut.service.UserServiceImpl.*(..))"/>

        <!--执行环绕增加!-->
        <aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
    </aop:config>

</beans>

测试类:

import com.demut.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        //动态代理 代理的是接口!!!
        UserService userService = context.getBean("userService", UserService.class);

        userService.select();
    }
}

运行结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-c40ZzuRq-1605142317547)(Spring%20%E9%9D%A2%E5%90%91%E5%88%87%E9%9D%A2%E7%BC%96%E7%A8%8B%EF%BC%88AOP%EF%BC%89%20D5.assets/image-20201111170324139.png)]

此方法中重点关注,applicationContext.xml中配置!

方法二(自定义类来实现)

创建diy包,在内部创建类:DiyPointCut,类中内容如下:

package com.demut.diy;

public class DiyPointCut {
    public void before() {
        System.out.println("************方法执行前************");
    }

    public void after() {
        System.out.println("************方法执行后************");
    }
}

修改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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--注册bean-->
    <bean id="userService" class="com.demut.service.UserServiceImpl"/>
    <bean id="log" class="com.demut.log.Log"/>
    <bean id="afterLog" class="com.demut.log.AfterLog"/>

    <!--方式二:自定义类-->
    <bean id="diy" class="com.demut.diy.DiyPointCut"/>
    <aop:config>
        <!--自定义切面ref要引用的类-->
        <aop:aspect ref="diy">
             <!--切入点-->
            <aop:pointcut id="point" expression="execution(* com.demut.service.UserServiceImpl.*(..))"/>
            <!--通知-->
            <aop:before method="before" pointcut-ref="point"/>
            <aop:after method="after" pointcut-ref="point"/>
        </aop:aspect>
    </aop:config>

</beans>

测试结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fyLTtQ7Z-1605142317549)(Spring%20%E9%9D%A2%E5%90%91%E5%88%87%E9%9D%A2%E7%BC%96%E7%A8%8B%EF%BC%88AOP%EF%BC%89%20D5.assets/image-20201111171851393.png)]

方法三(使用注解实现)

使用到的注解:

  • @Aspect
  • @Before
  • @After
  • @Around

在diy包下新建类: AnnotationPointCut类:

package com.demut.diy;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;

@Aspect //标注这个类是一个切面
public class AnnotationPointCut {
    @Before("execution(* com.demut.service.UserServiceImpl.*(..))")
    public void before() {
        System.out.println("==============方法执行前==============");
    }

    @After("execution(* com.demut.service.UserServiceImpl.*(..))")
    public void after() {
        System.out.println("==============方法执行后==============");
    }

    //在环绕增强中,我们可以给定一个参数,代表我们要获取处理切入的点
    @Around("execution(* com.demut.service.UserServiceImpl.*(..))")
    public void around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("环绕前");

        //获取签名~获取执行的是哪个类中的哪个方法
        Signature signature = jp.getSignature();
        System.out.println(signature);
        //执行方法
        Object proceed = jp.proceed();

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

在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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--注册bean-->
    <bean id="userService" class="com.demut.service.UserServiceImpl"/>
    <bean id="log" class="com.demut.log.Log"/>
    <bean id="afterLog" class="com.demut.log.AfterLog"/>

    <!--方式三:使用注解-->
    <bean id="annotationPointCut" class="com.demut.diy.AnnotationPointCut"/>
    <!--开启注解支持-->
    <aop:aspectj-autoproxy/>

</beans>

测试结果:

【参考文章】

https://www.cnblogs.com/xrq730/p/4919025.html

https://www.jianshu.com/p/994027425b44

写在最后

瀑布的水逆流而上,

蒲公英种子从远处飘回,聚成伞的模样,

太阳从西边升起,落向东方。

子弹退回枪膛,

运动员回到起跑线上,

我交回录取通知书,忘了十年寒窗。

厨房里飘来饭菜的香,

你把我的卷子签好名字,

关掉电视,帮我把书包背上

你还在我身旁。

​ ——戴畅

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值