Spring3之基于Aspect实现AOP

news2025/2/27 23:58:47

简介

使用 Aspect 搭配 Spring 可轻松实现 AOP;本章将通过一个完整示例演示如何实现这一功能

实现步骤

  1. 修改 beans.xml 配置文件的 schema 部分;可以在 spring-framework-reference.html 文件通过搜索关键字 “/aop” 找到配置 schema,然后把这三句 schema 添加到 beans.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: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-3.0.xsd
         http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context-3.0.xsd
         http://www.springframework.org/schema/aop
         http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">
      <!-- 设置Spring去哪些包中找annotation -->
   <context:component-scan base-package="com.ibm.oneteam"/>
</beans>
  1. 在 beans.xml 文件中添加如下基本配置
<context:component-scan base-package="com.duzimei.blog"/> <!-- 设置要被扫描的基础包 -->
<aop:aspectj-autoproxy/> <!-- 打开基于 Annotation 的AOP 设置 -->
  1. 添加如下 3 个依赖包到项目
    在这里插入图片描述

  2. 新建一个类 LogAspect.class, 添加如下代码


 // 声明这是一个切面类
public class LogAspect {
    // 在函数调用前执行
    ("execution(* com.duzimei.blog.dao.*.add(..) || "
        + "execution(* com.duzimei.blog.controller.*.delete(..))")
    public void logBefore(JoinPoint jp) {
        System.out.println("开始添加日志");
        // 输出执行对象
        System.out.println(jp.getTarget());
        // 输出执行方法
        System.out.println(jp.getSignature().getName());
    }
    
    // 在函数调用过程中执行
    ("execution(* com.duzimei.blog.dao.*.add(..)) || "
        + "execution(* com.duzimei.blog.controller.*.delete(..))")
    public Object logAround(ProceedingJoinPoint pjp) throws Throwable {
        System.out.print("1.开始添加日志");
        Object result = pjp.proceed();
        System.out.println("2.完成添加日志");
        return result;
    }
    
    // 在函数调用后执行
    ("execution(* com.duzimei.blog.*.add(..)) || "
        + "execution(* com.duzimei.blog.*.controller.delete(..))")
    public void logAfter() {
        System.out.println(“完成添加日志”);
    }
    
    // 函数调用出现异常
    (throwing="e", pointcut="execution(* com.duzimei.blog.*.add(..)) || "
        + "execution(* com.duzimei.blog.controller.*.delete(..))")
    public void logError(Throwable e) {
        System.out.println("添加日志失败, " + e.getMessage());
    }

    // 函数调用完毕后处理返回值
    (pointcut="execution(* com.duzimei.blog.*.add(..)) || "
        + "execution(* com.duzimei.blog.controller.*.delete(..))")
    public Object afterReturn() {
        System.out.println("获取到函数返回值");
        return result;
    }
}
  1. 测试一下;注意查看如下打印结果的执行顺序
    在这里插入图片描述

重要属性讲解

(1) @Before:在函数调用前执行

(2) @Around:在函数调用过程中执行,添加了该注解的方法是最主要的方法,基本上所有的逻辑都应在这个这个方法里

(3) @After:在函数执行完毕后执行

(4) @AfterThrowing:函数执行过程中出现异常;不常用,一般在 @Around 注解的方法中用 try … catch 处理较好

(5) @AfterReturning:函数执行完毕后执行,可获取、操作函数的返回值

(6) @Pointcut:用于定义切面,方便其他注解配置;如上面的例子中,每个方法的注解上都写了 execution 表达式,在实际开发过程中,可通过@Poingcut 定义不同的切面,然后在注解中直接调用,可简化代码;例如:

public class Pointcuts {
    // 定义只处理 dao 层 add 方法的切面配置
    ("execution(* com.duzimei.blog.dao.*.add(..))")
    public void logAdd() { }
    
    // 定义值处理 controller 层 delete 方法的切面配置
    ("execution(* com.duzimei.blog.controller.*.delete(..))")
    public void logDelete() { }
}

然后在 LogAspect.class 中可以这样使用



public class LogAspect {   
    ("com.duzimei.blog.aop.Pointcuts.logAdd()") 
    public void logBefore() {
        ... ...
    }
}

(7) execution 表达式:execution(* com.duzimei.blog.*.add(…)) 这个表达式中,第一个 * 表示任意返回值,第二个 * 表示 com.duzimei.blog 包路径下的所有类,第三个 * 表示以 add 开头的所有方法,(…) 表示方法的任意参数;多个表达式通过 || 分隔开

在 XML 文件中配置

上面演示的是通过直接在类中添加注解的方式实现 AOP,如果要使用 XML 的方式配置,只需添加如下代码即可

<bean id="logAspect" class="com.duzimei.blog.aop.LogAspect"/> <!-- 注入日志工具类 -->
<aop:config>
    <!-- 定义切面 -->
    <aop:aspect id="logAspect" ref="logAspect">
        <!-- 定义表达式和步骤 -->
        <aop:pointcut id="logPointcut" expression="execution(* com.duzimei.blog.dao.*.add(..))"/>
        <aop:before method="logStart" point-ref="logPointcut"/>
        <aop:after method="logAfter" point-ref="logPointcut"/>
    </aop:aspect>
</aop:config>

补充
在实际开发过程中,经常需要通过 AOP 实现日志记录功能,这里面稍微麻烦的地方在于如何获取方法的参数以及对应的值,因为我们在拦截的时候是不知道方法参数的类型的;这里有个简单的实现方式,参考代码如下

("execution(* com.dufu.blog.controller.*.*(..))")
public Object around(ProceedingJoinPoint point) throws Throwable {
    // 获取方法参数
    Object[] args = point.getArgs();
    // 获取方法签名
    MethodSignature signature = (MethodSignature)point.getSignature();
    // 获取执行对象
    Object target = point.getTarget();
    // 获取方法
    Method method = target.getClass().getMethod(signature.getName(), signature.getParameterTypes());
    // 如果方法上添加了 @Logger 注解, 记录用户操作
    if(method.isAnnotationPresent(Logger.class)) {
        LocalVariableTableParameterNameDiscoverer parameter = new LocalVariableTableParameterNameDiscoverer();
	// 获取方法参数名称
	String[] paramNames = parameter.getParameterNames(method);
	// 通过 Map 记录参数
	Map<String, Object> paramMap = new HashMap<>();
	if(null != paramNames && paramNames.length > 0) {
	    for(Integer i = 0; i < paramNames.length; i++) {
	        // 忽略类型是 request, response, multipartFile 的参数
		if(args[i] instanceof HttpServletRequest || args[i] instanceof HttpServletResponse 
                    || args[i] instanceof MultipartFile) {
		    continue;
		}
		paramMap.put(paramNames[i], args[i]);
	    }
	}
	// 打印一下参数信息
	System.out.println(JSON.toJSONString(paramMap));
    }
    return point.proceed();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/347226.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux安装ElasticSearch

下载地址&#xff1a;https://www.elastic.co/cn/downloads/past-releases#elasticsearch 1 版本选择 ElasticSearch 7 及以上版本都是自带的 jdk&#xff0c;假如需要配置指定的 jdk 版本的话&#xff0c;可以在 es 的 bin 目录下找到elasticsearch-env.bat 这个文件&#x…

(学习日记)2023.2.12

写在前面&#xff1a; 由于时间的不足与学习的碎片化&#xff0c;写博客变得有些奢侈。 但是对于记录学习&#xff08;忘了以后能快速复习&#xff09;的渴望一天天变得强烈。 既然如此 不如以天为单位&#xff0c;以时间为顺序&#xff0c;仅仅将博客当做一个知识学习的目录&a…

谷粒商城:订单中心概念解析

1、订单中心 电商系统涉及到 3 流&#xff0c;分别时信息流&#xff0c;资金流&#xff0c;物流&#xff0c;而订单系统作为中枢将三者有机的集 合起来。 订单模块是电商系统的枢纽&#xff0c;在订单这个环节上需求获取多个模块的数据和信息&#xff0c;同时对这 些信息进行加…

千峰jquery进阶内容

封装选项卡&#xff1a; HTML部分&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><meta name"viewport&…

在基于全志D1s的芒果派麻雀上运行国产开源rt-smart系统

想必RT-Thread系统大家不陌生了&#xff0c;RT-Thread Smart&#xff08;简称 rt-smart&#xff09;是基于 RT-Thread 操作系统衍生的新分支&#xff0c;面向带 MMU&#xff0c;中高端应用的芯片&#xff0c;例如 ARM Cortex-A 系列芯片&#xff0c;MIPS 芯片&#xff0c;带 MM…

ELK采集MySQL慢日志实现

文章目录一、ELK采集MySQL慢日志架构二、filebeat三、logstash四、eskibana一、ELK采集MySQL慢日志架构 MySQL 服务器安装 Filebeat 作为 agent 收集 slowLogFilebeat 读取 MySQL 慢日志文件做简单过滤传给 Kafka 集群Logstash 读取 Kafka 集群数据并按字段拆分后转成 JSON 格…

春季开学即将到来!大学生活必备数码清单奉上

马上就要开学了&#xff0c;你的返校装备是否已经准备齐全了呢&#xff1f;对于高校学生来说&#xff0c;很多数码产品都属于必备装备&#xff0c;比如下面这几款产品就受到了大量年轻消费者的喜爱&#xff0c;在它们的帮助下能够让大家的学习时光变得更快乐。1、不入耳黑科技骨…

DM8:DMDSC共享存储集群搭建-配置文件准备

DM8:DMDSC共享存储集群搭建-配置文件准备1 环境介绍2 配置文件2.1 EP73 节点配置文件2.1.1 dmdcr_cfg.ini2.1.2 dmasvrmal.ini2.1.3 dmdcr.ini2.1.4 dminit.ini2.1.5 dmcssm.ini2.1.6 配置文件就绪2.2 EP74 节点配置文件2.2.1 dmdcr.ini1 环境介绍 使用裸设备搭建 2 节点 DMDS…

从零开始的数模(二十六)单因素方差分析

目录 一、概念 1.1相关概念 1.2用途 1.3数据要求&#xff1a;独立性/正态性/方差齐性 1.4步骤 ​编辑1.5专业名词 二、基于python的单因素方差分析 2.2单因素方差分析的作用 一、概念 1.1相关概念 单因素方差分析是一种常用的统计分析方法&#xff0c;它用于比较一个因…

csgo搬砖项目,门槛最低的副业就是它(内附入门知识及选品技巧)

CSGO搬砖如何选择游戏饰品(装备&#xff09;&#xff1f;相信很多朋友一定很关心这个问题&#xff0c;因为如何选品直接关系到该装备是否快速出售&#xff0c;而且也关系到账号整体盈收状况。那么今天阿阳就来好好聊聊如何选择Steam装备以及饰品的各项知识点。 Steam搬砖如何选…

Leetcode力扣秋招刷题路-0061

从0开始的秋招刷题路&#xff0c;记录下所刷每道题的题解&#xff0c;帮助自己回顾总结 61. 旋转链表 给你一个链表的头节点 head &#xff0c;旋转链表&#xff0c;将链表每个节点向右移动 k 个位置。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], k 2 输出&…

linux另类提权之打靶归来(2019年老文)

0x00前言&#xff1a; 本次目标为渗透某安全培训学校在线靶场&#xff0c;朋友做不出来交帮忙看下。。。 不要听见靶场就关闭了此文&#xff0c;客官往下看。 0x01开端 打开此靶机各种琳琅满目的漏洞让我眼花缭乱&#xff0c;这里我选择一种直击要害的漏洞作为开端&#xff…

【入门】what is apache orc?提高hive存储效率?怎么和hive搭配使用?

文章目录一. 什么是orc二. hive集成orc三. hive表属性一. 什么是orc 官网&#xff1a;https://orc.apache.org/docs/ 1.ORC files目标为了提高hive的存储效率&#xff0c;以及减少文件大小。 Back in January 2013, we created ORC files as part of the initiative to massive…

转发一张网络工程师考试的试卷2021.5.15

网络工程师考试 单选题 &#xff08;30题&#xff0c;每题1分&#xff0c;共30分&#xff09; 1. 你在一个网络中实现DHCP服务&#xff0c;配置一些计算机成为DHCP客户端&#xff0c;由于工作需要&#xff0c;一台系统为Windows 10 的客户端要把从DHCP服务器获得的地址释放&a…

如何优化认知配比

战略可以归结为三种要素的合理配比。我们对战略的一个定义是&#xff1a;在终局处的判断。这其实来自于一个宗教的命题——面死而生。死是终局&#xff0c;生是过程&#xff0c;当你想做一个思想实验&#xff0c;或者是你真的有缘能够直面死亡&#xff0c;你所有关于生的认知就…

echonet-dynamic代码解读

1 综述 一共是这些代码&#xff0c;我们主要看echo.py&#xff0c;segmentation.py&#xff0c;video.py&#xff0c;config.py。 2 配置文件config.py 基于配置文件设置路径。 """Sets paths based on configuration files."""import conf…

大数据之-Nifi-了解Nifi处理器_和Nifi的其他组件---大数据之Nifi工作笔记0003

然后我们来看nifi的处理器,可以看到左上角是用来添加处理器的, 拖过来就能添加 拖过来以后,会显示一个弹框,里面会显示各种处理器,有293个..常用的都够了 可以在右边搜索以后添加 看一下常用的处理器 puthiveql:把数据写入到hive中去. 上面这3个是经常一块用的 这个publis…

Spring Boot邮件发送(powernode document)(内含源代码)

Spring Boot邮件发送&#xff08;powernode document&#xff09;&#xff08;内含源代码&#xff09; 源代码下载链接地址&#xff1a;https://download.csdn.net/download/weixin_46411355/87452780 目录Spring Boot邮件发送&#xff08;powernode document&#xff09;&…

Java企业级开发学习笔记

文章目录一、Spring1.1、Slay Dragon1.2、RescueDamselQuest一、Spring 第一周写了两个小项目均使用了原始调用和容器的方法 两个项目&#xff1a;<斩杀大龙与上路保卫战> 配一张文件位置图 1.1、Slay Dragon BraveKnight package net.sherry.spring.day01;public c…

卷积神经网络中的图像特征——以YOLOv5为例进行可视化

一、图像特征 1. 图像低层特征 图像低层特征指的是&#xff1a;边缘、颜色和纹理等特征。低层特征的分辨率较高&#xff0c;包含较多的位置、细节信息&#xff0c;但其包含的语义信息较少&#xff0c;噪声较多。原始图像和浅层卷积网络输出的特征图属于低层特征&#xff0c;从…