AOP切面编程 -- Spring信息流控制的核心

2024-05-05 13:48

本文主要是介绍AOP切面编程 -- Spring信息流控制的核心,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

        Spring AOP被称为面向切面编程。

       它可以在某一个或者一类方法调用的前后添加其他的处理方法与逻辑对参数或者流程进行修改,提供对同一类型处理进行批量改动的能力

       同时,也可以实现修改第三方class文件已编写好的类、jar包等无法直接编辑源码但又想做改动的功能。


一. 配置

       首先,需要再Spring maven项目中添加aspect的引用jar包

		<!-- aop切面编程包 --><dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.8.10</version></dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.10</version></dependency>

          在Spring.xml配置文件中,添加下面的配置

<beans xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd"><!-- 开启aop注解 @Aspect --><aop:aspectj-autoproxy/></beans>

之后就可以使用aspect编程了

二. 编程

           使用@Aspect注解,编写一个切面bean

//使用@Aspect注解使此类成为一个aspect类
@Aspect
@Component        //本身aspect类也得是个bean在Spring注册启动,否则不会生效
public class ServiceAspect {//定义一个切点,下面可以直接用这个切点作为连接点,在它前后进行切面编程操作@Pointcut("execution(** serviceMethod1(..)) ")public void method1(){}@Pointcut("execution(** serviceMethod2(String)) ")public void method2(){}@Before("method1()")			                    //使用切点public void before(){System.out.println("AOP serviceMethod1 BEFORE reach");}@Before("method2() && args(param)")				//调用有参数方法public void before2(String param){System.out.println("AOP BEFORE2 reach");System.out.println("AOP param=="+param);}@Around("method2()")public String roundAspect(ProceedingJoinPoint jp){String rString = "";try{System.out.println("round before Join Point");rString = (String) jp.proceed();	//调用方法,需要将方法的调用值返回,否则controller调用service得到的结果为nullSystem.out.println("round after Join Point");} catch(Throwable exception) {System.err.println("ASPECT error");rString = "error";}return rString;}//类方法拓展,将MyService接口实现类扩展为实现了ExtendInterface接口的ExtendClass类的实例,使用时强转成ExtendClass类型就可以@DeclareParents(value = "com.service.MyService+",defaultImpl=ExtendClass.class)public ExtendInterface extendInterface;							}

         连接点是serviceMethod1()和serviceMethod2(String)两个方法,定义在接口MyService中

public interface MyService {public String serviceMethod1();public String serviceMethod2(String param);
}

实现类 

@Service()	
public class MyServiceImpl implements MyService {@Overridepublic String serviceMethod1() {		return "method1";}@Overridepublic String serviceMethod2(String param) {		return "method2,  param=="+param;}
}

           则在切入点的方法被调用前后,Aspect切面里面的相关的日志就会打印。

三. 切入类型

1. @Before

  • 在切点方法调用前执行
  • value值:切入点规则表达式
  • @Before执行完后切点方法会自动执行,除非@Before抛异常
  • @Before方法无法访问目标返回值

2. @AfterRunning

  • 在目标方法正常完成后执行
  • value/pointcut:切入点规则表达式
  • returning:该属性指定一个形参名,方法可以定义一个同名的形参,来接收目标方法的返回值
  • returning可以限制返回值类型(相同类型或void没有返回值),辅助value属性进行方法筛选
  • @AfterRunning无法修改最终的返回值

3. @AfterThrowing

  • 处理程序中抛出的异常
  • value/pointcut:切入点规则表达式
  • throwing:指定一个形参名,方法可定义同名形参,用于接收抛出的异常Exception
  • @AfterThrowing无法完全处理异常。即使进行处理,切入点方法的原先调用上层仍然会得到抛出的异常

4. @After

  • @AfterRunning相似
  • @AfterRunning不同的是,不管切入方法是否正常完成,都会进入@After方法
  • @After方法最好分为两部分,正常处理和异常处理,并且在异常处理内释放资源

5. @Around

  • 近似于@Before处理和@AfterRunning处理总和
  • 可以决定在哪里进行前后增强操作,甚至组织目标方法执行(不调用)
  • 可以改变入参和返回值
  • 需要在线程安全的情况下使用
  • 第一个参数必须是ProceedingJoinPoint类型,并且必须调用proceed()方法才能执行切入点的方法,否则切入点方法不执行
  • 切入点方法的原先调用上层得到的返回值是Around的返回值。因此必须将proceed()方法结果返回,原先调用上层才能获取到切入点的返回值

得到入参:

Object[] args = jp.getArgs() ;

 

四. 切入点表达式

表达式知识可以参考文章《spring AspectJ的Execution表达式-备忘笔记》

总结下最常用的execution()切点函数

execution() 用于匹配执行方法的连接点,语法规则如下

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)

第一个是修饰符(modifiers-pattern) :public、protected、private
第二个是返回值(ret-type-pattern) :  不可为空,最少为*
第三个方法的包名(declaring-type-pattern)  
第四个是方法名(name-pattern) :不可为空,最少为*
第五个方法参数(param-pattern) : 不可为空,任意类型用".."表示
第六个抛出的异常类型(throws-pattern)

因此,最简单的表达式:execution("* *(..)")  表示任意方法

这篇关于AOP切面编程 -- Spring信息流控制的核心的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/961880

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2