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编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("