详解原生Spring当中的额外功能开发MethodBeforeAdvice与MethodInterceptor接口!

 😉😉 学习交流群:

✅✅1:这是孙哥suns给大家的福利!

✨✨2:我们免费分享Netty、Dubbo、k8s、Mybatis、Spring...应用和源码级别的视频资料

🥭🥭3:QQ群:583783824   📚📚  工作微信:BigTreeJava 拉你进微信群,免费领取!

🍎🍎4:本文章内容出自上述:Spring应用课程!💞💞

💞💞5:以上内容,进群免费领取呦~ 💞💞💞💞

一:额外功能

1:MethodBeforeAdvice

        额外功能实现了MethodBeforeAdvice接口,实现这个接口就需要实现这个接口当中的before方法,他的作用就是让额外功能运行在原始方法之前,这就是这个before方法的作用,before方法里边有着三个参数,method,Object[],Object,第一个参数method代表的就是要增加的目标方法或者叫原始方法,如何此时要增加的方法是login方法,那么Method代表的就是login方法,这个Method是变化的,Object[]代表的是原始方法的参数,Object对应的是原始对象,代表的是原始类或者目标类的实例。我们怎么证明这个事呢?可以通过debug来进行测试

        methodbeforeAdvice这个接口核心的作用就是保证。额外功能运行在原始方法执行之前,进行额外功能操作。

        MethodBeforeAdvice中before方法中的三个参数如何使用呢?
        作为一个接口所规定的接口方法参数我们不一定都得用,根据实际情况进行使用就行。举例来讲:前面我们在学servlet的时候,servlet规定了一个方法,service方法,这个方法有两个参数httpRequest,httpResponse这两个参数,servlet这个接口为我们提供了这样的方法,在我们后续的使用的过程当中,我们没有使用过后边的参数。

        例如我们只用来接收客户端的请求参数,直接使用request这个参数就好了,如果我们使用servlet是进行页面响应的,那么我们使用response这个参数就好了,使用这个参数获取输出流输出流进行页面响应。通过before方法的参数在实战中,会根据需要进行使用。

2:MethodInterceptor

        MethodInterceptor 方法拦截器。这个接口是完成额外功能实现的时候的第二个接口。这个接口和MethodBeforeAdvice有什么区别呢?MethodBeforeAdvice只能运行在原始方法之前,相对来讲,功能单一。

       MethodInterceptor  这个的接口的实现类的方法可以运行在原始方法执行之前,也可以运行在执行方法的执行之后,这个功能更加的强大一点,所以在实际过程中我们更多的使用的是MethodInterceptor这个接口,下面的这个:

/*** @Auther: DaShu* @Date: 2021/6/21 20:12* @Description:*/
public class Around implements MethodInterceptor {@Override//这个接口的方法是有返回值的这也就解释了spring的注解开发的aop是Around方法里边的返回值是Object类型的//这个接口里边有一个invoke方法,我们也就把额外功能写入invoke当中,这个额外功能就能运行在原始方法之前,也能运行在原始方法之后。也可以运行在原始方法之前,之后。//原始方法怎么运行呢?这就需要知道这个方法的参数的含义://这个方法的参数:这个就代表的事原始方法,底层相当于是对目标方法对象Method的一个更高级别的封装,可以认为他们两个设计相等的。methodInvocation.proceed()方法就代表目标方法执行了。//这个MthodInvocation就代表的事目标方法,而这里的methodinvocation.invoke就代表目标方法的运行。//到这:讲清楚了原始方法的运行是怎么回事的,//这个方法的返回值:这个返回值的原因在于实现接口的时候,接口按照接口返回值的类型,invoke方法有返回值 ,这里的Object//代表的就是原始方法执行之后的返回值,所以这个方法必须有一个Object,如果原始方法的返回值是void,那么我们返回一个null就行了。public Object invoke(MethodInvocation methodInvocation) throws Throwable {return null;}
}
<?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:util="http://www.springframework.org/schema/util"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.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd"><bean id="userService" class = "com.pactera.spring.proxy.UserServiceImpl"/><bean id="orderService" class = "com.pactera.spring.proxy.OrderServiceImpl"/>
<!--    <bean id="before" class="com.pactera.spring.dynamic.Before"/>--><bean id = "around" class = "com.pactera.spring.dynamic.Around"/><aop:config><!--这个标签就是用来定义切入点的,expression这个是切入点表达式,代表所有的方法都要加上这个额外功能--><!--所有的方法,都作为切入点作为额外功能。--><aop:pointcut id="pc" expression="execution(* *(..))"/><!--组装,组长的目的就是为了把切入点与额外的功能进行整合--><aop:advisor advice-ref="around" pointcut-ref="pc"/></aop:config>
</beans>
  /** @Target: 用于测试Arround方法* @Author: DaShu* @Date: 2021/6/21 20:53* @Result:*/@Testpublic void test3(){ApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext3.xml");com.pactera.spring.proxy.OrderServcie orderService = (com.pactera.spring.proxy.OrderServcie)ctx.getBean("orderService");orderService.showOrder();//...........................这里是额外功能...............................//OrderServiceImpl.showOrder//-------------------------------这里还是额外功能----------------------------------------}


        总结:MehtodInterceptor这个接口的额外功能就是实现这接口的类被Spring创建的对象会被视为代理对象,实现并重写的方法invoke方法中书写额外功能,额外功能可以在目标方法前执行,可以在目标方法后执行,这是一种环绕执行的手段,很常用。

        总结2:让额外功能运行在原始方法执行之后执行,这就是废话。
        总结2:让额外功能运行在原始方法执行之前后执行,这也是废话。
        总结3:什么样的额外功能在方法的运行之前和运行之后都要添加呢?
       (日志,事务,性能分析等等。如图:

        还有一个额外功能不拘于之前之后,是运行在原始方法抛出异常的时候。

        额外功能执行在原始方法抛出异常的时候,这样的话,我们就在原始方法抛出异常的时候,在这个invoke当中的proceed方法周围进行trycatch,将异常进行捕获,捕获之后,将额外功能写到catch的代码快中就行了。

package com.pactera.spring.dynamic;import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;/*** @Auther: DaShu* @Date: 2021/6/21 20:12* @Description:*/
public class Around implements MethodInterceptor {@Override//这个接口的方法是有返回值的这也就解释了spring的注解开发的aop是Around方法里边的返回值是Object类型的//这个接口里边有一个invoke方法,我们也就把额外功能写入invoke当中,这个额外功能就能运行在原始方法之前,也能运行在原始方法之后。也可以运行在原始方法之前,之后。//原始方法怎么运行呢?这就需要知道这个方法的参数的含义://这个方法的参数:这个就代表的事原始方法,底层相当于是对目标方法对象Method的一个更高级别的封装,可以认为他们两个设计相等的。methodInvocation.proceed()方法就代表目标方法执行了。//这个MthodInvocation就代表的事目标方法,而这里的methodinvocation.invoke就代表目标方法的运行。//到这:讲清楚了原始方法的运行是怎么回事的,//这个方法的返回值:这个返回值的原因在于实现接口的时候,接口按照接口返回值的类型,invoke方法有返回值 ,这里的Object//代表的就是原始方法执行之后的返回值,所以这个方法必须有一个Object,如果原始方法的返回值是void,那么我们返回一个null就行了。public Object invoke(MethodInvocation methodInvocation) throws Throwable {System.out.println("...........................这里是额外功能...............................");Object proceed = null;try {proceed = methodInvocation.proceed();} catch (Throwable throwable) {throwable.printStackTrace();System.out.println("-------------------------------这里还是核心功能------------------------");}System.out.println("-------------------------------这里还是额外功能----------------------------------------");return proceed;}
}


        MethodInterceptor中的可以以影响原始方法中的返回值,我们在添加功能的时候我们可以获取到原始方法的返回值,然后我们可以把原始方法的返回值进行修改,这样就达到了修改的目的。
        其实你看Spring的这个Aop确实挺强大的,但是呢,前提是你得按照他的Aop的4个方法和步骤进行实现才行,最基本的附加功能类和目标类必须得是Spring进行创建的吧,切入点也得定义把,然后也得组装把,只有这样才能搞出来这个Aop。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux 磁盘扩容初始化挂载 笔记

目录 说明环境信息前提条件 操作步骤 说明 linux 系统磁盘扩容步骤 环境信息 系统信息&#xff1a;Linux version 4.19.90-23.8.v2101.ky10.aarch64cpu信息&#xff1a;Kunpeng-920 、aarch64、64-bit、HiSilicon 前提条件 有未初始化的用户磁盘操作系统可以支持当前磁盘的…

如何快速生成项目目录结构树?

经常在网上看到下面这种由一个项目&#xff0c;生成一个结构树&#xff0c;你知道它是怎么生成的吗&#xff1f; 这就是利用本文要介绍的一个工具——Treer&#xff0c;treer就是一款专门用来快速生成目录结构树的命令行工具。 第一步&#xff1a;安装treer 在终端执行全局…

前端编码规范

文章目录 一、背景二、内容1、注释规范&#xff08;1&#xff09;文件注释&#xff08;2&#xff09;函数注释&#xff08;3&#xff09;单行注释&#xff08;3&#xff09;多行注释 2、命名规范&#xff08;1&#xff09;项目命名&#xff08;2&#xff09;目录命名&#xff0…

[ffmpeg] aac 音频编码

aac 介绍 aac 简单说就是音频的一种压缩编码器&#xff0c;相同音质下压缩比 mp3好&#xff0c;目前比较常用。 aac 编码支持的格式 aac 支持的 sample_fmts: 8 aac 支持的 samplerates: 96000 88200 64000 48000 44100 32000 24000 22050 16000 12000 11025 8000 7350 通…

uniapp开发小程序使用axios进行网络请求 uniapp 小程序调试

前言 本篇最好放到项目的【README.md】文件中,方便每次发布的时候检查纠错,毕竟好记性不如烂笔头。而且其他开发者帮忙修改bug、发布新版本的时候,只需要根据这个事项就能实现整个流程的提审发布,提高效率。 1、微信小程序配置 1.1、检查APPID是否正确 测试:wx--------…

Day45力扣打卡

打卡记录 无矛盾的最佳球队&#xff08;线性DP&#xff09; class Solution:def bestTeamScore(self, scores: List[int], ages: List[int]) -> int:n len(scores) nums sorted(zip(scores, ages))f sorted(scores)for i in range(n):for j in range(0, i):if nu…

jdk动态代理和CGLIBE代理

静态代理&#xff1a;由程序员创建或特定工具自动生成源代码&#xff0c;再对其编译。在程序运行前&#xff0c;代理类的.class文件就已经存在了。 动态代理&#xff1a;在程序运行时&#xff0c;运用反射机制动态创建而成。 使用jdk的反射机制&#xff0c;创建对象的能力&…

【Vulnhub靶机】lampiao--DirtyCow

文章目录 漏洞介绍简介原因类型版本危害 信息收集主机扫描端口扫描 漏洞探测漏洞利用权限提升nc文件传输编译 参考 靶机地址&#xff1a;lampiao 下载地址&#xff1a;Lampio: 1 漏洞介绍 简介 脏牛&#xff08;Dirty Cow&#xff09;是Linux内核的一个提权漏洞&#xff0c;…

Grafana部署与Zabbix集成,搭建开源IT监控平台

Grafana部署与Zabbix集成 目前在一家公司主要是网络、运维、IT支持&#xff0c;每次需要检查服务器状态都是需要手动登录系统进行查看&#xff0c;因此想着部署一套监控系统&#xff0c;功能上需要实现监控、可视化、告警等。由于预算没有&#xff0c;服务器资源倒是有空闲的&a…

通过流量监控分析某个部门或客户端网络性能

在当今数字化时代&#xff0c;网络已经成为组织和企业不可或缺的基础设施之一。作为信息传输和数据交互的关键载体&#xff0c;网络的性能对于保障业务的稳定运行和提升工作效率至关重要。因此&#xff0c;对某个部门或客户端网络的性能进行分析和评估&#xff0c;有助于了解当…

【开源视频联动物联网平台】视频AI智能分析部署方式

利用视频监控的AI智能分析技术&#xff0c;可以让视频监控发挥更大的作用&#xff0c;成为管理者的重要决策工具。近年来&#xff0c;基于视频监控的AI分析算法取得了巨大的发展&#xff0c;并在各种智慧化项目中得到了广泛应用&#xff0c;为客户提供更智能化的解决方案。 然…

Android 10.0 Launcher3定制之首页时钟小部件字体大小的修改

1.前言 在10.0的产品开发中,在一些Launcher3的定制化开发中,在对于一些小屏幕的产品开发中,在首页添加时钟小部件会显得字体有点小, 所以为了整体布局美观就需要改动小部件的布局日期字体的大小来实现整体的布局美观效果,接下来来具体实现相关的功能 具体效果图: 2.Lau…