对Spring当中AOP的理解

        AOP(面向切面编程)全称Aspect Oriented Programminge

        AOP就是把系统中重复的代码抽取出来,单独开发,在系统需要时,使用动态代理技术,在不修改源码的基础上,将单独开发的功能通知织入(应用)到系统中的过程,完成完整的代码逻辑

        优点:减少重复代码、提高开发效率、提高维护性

AOP相关术语:

        1. Joinnoint(连接点):连接点是指那些被拦截到的点。这些点指的是方法,spring只支持方法类型的连接点。指切面在系统中要应用的位置,spring 中能够使用切面的位置为方法,所以连接指方法连接点指要调用某个方法时拦截到对这个方法的调用,做一些其他事情然后再执行目标方法拦截到的方法就是连接点。

        2. Pointcut(切入点):指我们要对那些Joinpoint进行拦截的定义,spring中使用切入点表达式定义(切入点表达式)简单理解,切入点是对连接点的描述,它将多个连接点进行概括的定义。

        3. Advice(通知):指拦截到joinpoint之后所要做的事情就是通知

         通知:前置通知、后置通知、异常通知、最终通知、环绕通知

        4. Target(目标对象):代理的目标对象

        5. Weaving(织入):将通知应用到切入点的过程

        6.Proxy(代理) :代理对象 

        7. Aspect(切面) :切入点和通知的总称

         简单说AOP是,在系统执行某个方法时,在执行这个方法前或方法后去执行另一段程序(另一段程序就是AOP中的通知),而拦截这个方法的定义可以称为切入点

        

SprinaAOP.的使用步骤(基于注解):
        1.引入spring依赖包和aop.植入依赖包 Aspect Weaver包或者spring-aspects包:

         2.创建切面类
        3.将切面类交个spring进行管理,使用@Component注解
        4.将切面类设置为一个切面,使用@Aspect注解
        5.编写切入点poincut,使用切入点表达式定义连接点(方法)。
        Execution(表达式)
        表达式写法:访问修饰符返回类型包名.包名.类名.方法名(参数列表)//指定那些方法做为切入点(拦截那些方法)

使用具体方法方式:@PointCut( execution(* com.xxx.service.UserService.addUser(..)))

了解更多Execution表达式Introduction to Pointcut Expressions in Spring | Baeldung

        6.编写相关通知

        7.切面源码

       

            在看切面源码前,我们需要提前把所需的环境先准备好,这里看一下我的结构:

        

这块我直接用service包下的UserService接口的addUser()和它的实现类UserServiceImpl来进行体现上述Advice(通知).

先看UserService接口

package com.xxx.service;

public interface UserService {

    public String addUser(int id,String name);

    public void delUser();
}

UserServiceImpl

package com.xxx.service.impl;

import com.xxx.service.UserService;
import org.springframework.stereotype.Service;

@Service
public class UserServiceImpl implements UserService {
    @Override
    public String addUser(int id,String name) {
        System.out.println("UserServiceImpl--------addUser");
        return "你好:"+name;
    }
    @Override
    public void delUser() {
        System.out.println("UserServiceImpl--------delUser");
    }
}

下面我们来看最主要的aspects包下的UserAspects类,该类就是对上述addUser()方法进行前置通知、后置通知、异常通知、最终通知、环绕通知。

首先咱们先看前置通知:

在前置通知前首先我们执行一下程序,结果如下:

然后我们加上前置通知代码后,再看结果:

package com.xxx.aspects;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;

@Aspect       //切面注解
@Component    //交给Spring进行控制翻转
@EnableAspectJAutoProxy  //开启spring的切面自动代理功能
public class UserAspects {


  @Pointcut("execution(* com.xxx.service.UserService.addUser(..))")
    public void pointCuts() {
    }

   /*前置通知*/
    @Before("pointCuts()")
    public void beforeAdvice(){
        System.out.println("前置被通知执行");
    }
}

 再加上前置通知后,结果如下:

下面我们来看后置通知,通过上述的前置通知,我们可以推断出后置通知应该就是addUser()执行后进行通知的,下面我们来看看是不是这样的。首先和之前一样,加上后置通知的代码:

package com.xxx.aspects;


import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;

@Aspect       //切面注解
@Component    //交给Spring进行控制翻转
@EnableAspectJAutoProxy  //开启spring的切面自动代理功能
public class UserAspects {


  @Pointcut("execution(* com.xxx.service.UserService.addUser(..))")
    public void pointCuts() {
    }


    /*前置通知*/
    @Before("pointCuts()")
    public void beforeAdvice(){
        System.out.println("前置被通知执行");
    }

    /*后置通知*/
    @AfterReturning("pointCuts()")
    public void AfterAdvice(){
        System.out.println("后置被通知执行");
    }
}

加上后置通知后,运行结果如下:

下面我们来看异常通知,字面意思就是,在程序出现异常的时候就会出现该通知,下面我们就看相关的代码:

package com.xxx.aspects;


import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;

@Aspect       //切面注解
@Component    //交给Spring进行控制翻转
@EnableAspectJAutoProxy  //开启spring的切面自动代理功能
public class UserAspects {


  @Pointcut("execution(* com.xxx.service.UserService.addUser(..))")
    public void pointCuts() {
    }


    /*前置通知*/
    @Before("pointCuts()")
    public void beforeAdvice(){
        System.out.println("前置被通知执行");
    }

    /*后置通知*/
    @AfterReturning("pointCuts()")
    public void AfterAdvice(){
        System.out.println("后置被通知执行");
    }
    /*异常通知*/
    @AfterThrowing("pointCuts()")
    public void throwAdvice(){
        System.out.println("异常通知执行");
    }
}

由于异常通知是在程序出现异常的时候才会执行该通知,那么咱们就给addUser()中来个异常

下面我们来看运行结果:

这儿需要注意的是异常通知和后置通知只能执行其中一个,通知的执行过程类似于下面这种:

下面的最终通知就不写了,它前面所述的其它通知类似,环绕通知就是一个可以做它们4个做的事情,下面直接把源码给出来看一下就行了:

        

package com.xxx.aspects;


import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;

@Aspect       //切面注解
@Component    //交给Spring进行控制翻转
@EnableAspectJAutoProxy  //开启spring的切面自动代理功能
public class UserAspects {


  @Pointcut("execution(* com.xxx.service.UserService.addUser(..))")
    public void pointCuts() {
    }


    /*前置通知*/
    @Before("pointCuts()")
    public void beforeAdvice(){
        System.out.println("前置被通知执行");
    }

    /*后置通知*/
    @AfterReturning("pointCuts()")
    public void AfterAdvice(){
        System.out.println("后置被通知执行");
    }

    /*异常通知*/
    @AfterThrowing("pointCuts()")
    public void throwAdvice(){
        System.out.println("异常通知执行");
    }
    /*最终通知*/
    @After("pointCuts()")
    public void finallyAdvice(){
        System.out.println("最终通知执行");

    }

    /*
    环绕通知就是可以把上面四个通知都可以实现
    * */
    @Around("pointCuts()")
    public String aroundAdvice(ProceedingJoinPoint proceedingJoinPoint){
        try {
            System.out.println("环绕通知----前置被通知执行");
            String proceeds = (String) proceedingJoinPoint.proceed();
            return proceeds;
        } catch (Throwable e) {
            e.printStackTrace();
        }
        return null;
    }

}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

shopee,lazada卖家自养号测评补单的方法和技巧

现在很多卖家都是自己管理几百个账号,交给服务商不是特别靠谱 一:送测不及时,产品时常送不出去 二:账号质量不稳定,账号一天下了多少你也不清楚,如果下了很多单万一封号被关联了怎么办 三:as…

ESP8266采用AT指令连接华为云服务器方法(MQTT固件)

一、前言 本篇文章主要介绍3个内容: (1)ESP8266-WIFI模块常用的AI指令功能介绍 (2)ESP8266烧写MQTT固件连接华为云IOT服务器。 (3)介绍华为云IOT服务器的配置过程。 ESP8266是一款功能强大…

vite引入图片用法

在 vite 中 引入图片方式跟其他脚手架创建项目的引入方式不一样 要使用 import.meta.url 他是一个 ESM 的原生功能 const mapPicSrc ref(new URL(/assets/charts/bdf.png, import.meta.url).href) 如果使用 require 引入会报错 说没有 require 模块

【Shell实战案例面试题】输入网卡的名字,来输出网卡的IP

1.问题 参数后判断要加"" 名字为空时显示ip 2.分析 把本机的所有网卡名列出来,来引导用户输入 使用命令列出所有网卡信:ifconfig/ip a 设计一个函数,把网卡名作为参数,函数返回网卡的IP 在获取某个网卡IP时,考虑网…

解决NuxtJS3中安装pinia报错

使用npm install pinia 安装pinia报错。 修改镜像后也报错: 方式一: npm config set registry https://registry.npm.taobao.org/ npm install pinia方式二: npm config set registry https://registry.npmjs.org/ npm install pinia解决方…

C++仿函数、万能头文件、transform学习

这是网上的一个代码,里面的一些东西以前没用过; #include <bits/stdc++.h> using namespace std;// A Functor class increment { private:int num; public:increment(int n) : num(n) { }int operator () (int arr_num) const {return num + arr_num;} };// Driver …

关于TongWeb部署应用报错java.lang.NoSuchFieldError: REFLECTION (by liuhui)

关于TongWeb部署应用报错java.lang.NoSuchFieldError: REFLECTION &#xff08;by liuhui&#xff09; 关于TongWeb部署应用报错java.lang.NoSuchFieldError: REFLECTION 问题现象&#xff1a;xml解析对象工厂类错误导致解析失败 解决办法&#xff1a;增加配置参数问题解决 -…

awk命令使用方法

简介 awk 是一种强大的文本处理工具&#xff0c;可以用于处理结构化的文本数据。它可以根据指定的模式和动作来筛选、处理和格式化文本。 下面是一些常见的 awk 命令使用方法。 详细介绍 基本语法&#xff1a; awk pattern { action } filename其中&#xff0c;pattern 是用…

【LeetCode: Z 字形变换 + 模拟】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

(自用)learnOpenGL学习总结-高级OpenGL-帧缓冲Framebuffers

我们在之前使用了很多缓冲了&#xff1a;颜色缓冲、深度缓冲、模板缓冲。这些缓冲结合起来叫做帧缓冲&#xff0c; 其实也能从名字理解&#xff0c;每一帧屏幕都需要不断更新画面&#xff0c;对应的缓冲也需要更新。 不过上面这些都是在默认的缓冲里面做的&#xff0c;现在我…

数据建模及可视化解决方案

一、需求背景 1. 数据量的爆炸性增长:随着互联网和物联网技术的发展,我们正处于一个数据爆炸的时代。企业和组织需要有效地管理和利用这些海量数据,而数据建模平台可以帮助他们实现这一目标。 2. 数据多样性和复杂性:数据来源的多样化和复杂性使得数据管理和分析变得更加…

基于 GPU 渲染的高性能空间包围计算

空间包围检测在计算机图形学、虚拟仿真、工业生产等有着广泛的应用。 现代煤矿开采过程中&#xff0c;安全一直是最大的挑战之一。地质空间中存在诸多如瓦斯积聚、地质构造异常、水文条件不利等隐蔽致灾因素&#xff0c;一旦被触发&#xff0c;可能引发灾难性的后果。因此在安…

架构整洁之道-组件构建原则

5 组件构建原则 大型软件系统的架构过程与建筑物修建很类似&#xff0c;都是由一个个小组件组成的。所以&#xff0c;如果说SOLID原则是用于指导我们如何将砖块砌成墙与房间的&#xff0c;那么组件构建原则就是用来指导我们如何将这些房间组合成房子的。 5.1 组件 组件是软件的…

漏洞原理SQL注入 手工注入漏洞

漏洞原理SQL注入 手工注入漏洞 SQL 注入的前置知识 information_schema库information_schema 是mysql5.0以上版本中自带的一个数据库。tables表information_schema库中的tables表中table_schema列&#xff08;存储数据库名&#xff09;和table_name列&#xff08;存储表名&…

慎投!这3本期刊诚信风险高,被警告和风险低的期刊选哪个?

近年来&#xff0c;学术界对学术诚信的关注越来越多&#xff0c;期刊的学术诚信风险也日益受到关注。科睿唯安在今年三月份剔除的35本SCI期刊中&#xff0c;绝大多数涉及学术诚信问题&#xff01;据官方介绍&#xff0c;其已正式开发了一款AI工具&#xff0c;可以锁定存在学术诚…

R-YOLO

Abstract 提出了一个框架&#xff0c;名为R-YOLO&#xff0c;不需要在恶劣天气下进行注释。考虑到正常天气图像和不利天气图像之间的分布差距&#xff0c;我们的框架由图像翻译网络&#xff08;QTNet&#xff09;和特征校准网络&#xff08;FCNet&#xff09;组成&#xff0c;…

ES 分词器

概述 分词器的主要作用将用户输入的一段文本&#xff0c;按照一定逻辑&#xff0c;分析成多个词语的一种工具 什么是分词器 顾名思义&#xff0c;文本分析就是把全文本转换成一系列单词&#xff08;term/token&#xff09;的过程&#xff0c;也叫分词。在 ES 中&#xff0c;Ana…

部署一个私有化的博客系统

效果 安装 1.创建目录 mkdir /opt/typecho/usr cd /opt/typecho 2.编写配置文件 vim docker-compose.yml 内容如下 version: 3 services:typecho:image: joyqi/typecho:nightly-php7.4-apachecontainer_name: typechorestart: alwaysenvironment:- TYPECHO_SITE_URLhttp:…

大模型应用开发:为产品创建一个AI客服/智能助手

欢迎阅读本系列文章&#xff01;我将带你一起探索如何使用OpenAI API来开发GPT应用。无论你是编程新手还是资深开发者&#xff0c;都能在这里获得灵感和收获。 本文将继续展示AI助手的开发方式&#xff0c;在OpenAPI中它的名字是Assistants。 什么是Assistants&#xff1f; 在之…

《信息系统项目管理师》备考计划

《信息系统项目管理师》证书价值和作用 信息系统项目管理师证书的价值体现在提升专业知识、助力职业发展、职称评定、技能补贴和人才引进或积分落户等方面。 一方面可以通过系统学习&#xff0c;可以显著提高自己在信息系统项目管理领域的专业水平。方便自己以后向更多的方向…