SpringBootWeb 篇-深入了解 AOP 面向切面编程与 AOP 记录操作日志案例

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍

文章目录

        1.0 AOP 概述

        1.1 构造简单 AOP 类

        2.0 AOP 核心概念

        2.1 AOP 执行流程

        3.0 AOP 通知类型

        4.0 AOP 通知顺序

        4.1 默认按照切面类的类名字母排序

        4.2 用 @Order(数字) 注解加在切面类上来控制顺序

        5.0 AOP 切入点表达式

        5.1 使用 execution() 创建切入点表达式

        5.2 使用 @annotation 创建切入点表达式

        6.0 AOP 连接点

        7.0 AOP 案例 - 记录操作日志


        1.0 AOP 概述

        AOP,Aspect Oriented Programming 面向切面编程,在 AOP 中,横切关注点被称为切面(Aspect),切面通过特定的注入方式被应用到程序的不同部分,从而实现对这些部分的增强或修改。AOP 能够帮助开发者更好地管理程序的复杂性,提高代码的重用性和易读性。

        简单来说,就是面向特定的方法编程,也或者说给原始的方法进行升级改造。这样原始的方法就不需要进行改变,从而实现方法升级了。如日志记录、权限控制等功能。通过AOP,可以实现方法的升级改造,提高代码的可维护性和可重用性。

        1.1 构造简单 AOP 类

        1)首先导入 AOP 依赖:

        在 pom.xml 中导入 AOP 的依赖。

    <!--AOP依赖--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId></dependency>

        2)编写 AOP 程序:

        针对特定的方法业务需要进行编程。

        首先创建类,在类上加上 @Component 注解进行控制反转,成为 IOC 容器中的 Bean 对象。继续在类上加上 @Aspect 注解,代表当前类不是普通类而是 AOP 类。在方法上加上通知类型,根据切入点表达式来筛选出连接点。

代码演示:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;@Component
@Aspect
public class demo1 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行目标代码之前的代码");Object result = joinPoint.proceed();System.out.println("正在执行目标代码之后的代码");return result;}
}

        2.0 AOP 核心概念

        连接点:JoinPoint,可以被 AOP 控制的方法(暗含方法执行时的相关信息)。

        通知:Advice,指哪些重复的逻辑,也就是共性功能(最终体现为一个方法)。

        切入点:PointCut,匹配连接点的条件,通知仅会在切入点方法执行时被应用。

        切面:Aspect,描述通知与切入点的对应关心(通知+切入点)。

        目标对象:Target,通知所应用的对象。

连接点与切入点的区别:

        简而言之,连接点是具体的程序执行事件,而切入点是一种筛选连接点的机制,可以帮助我们选择在哪些连接点应用切面逻辑。对于切入点来说,是通过切入点表达式来描述切入点。

        2.1 AOP 执行流程

        1)首先定义切面:

        开发人员定义一个切面,包含通知和切入点的定义。通知定义了切面逻辑,包括前置通知、后置通知、环绕通知等。切入点定义了在哪些连接点上应用切面逻辑。

        2)创建目标对象和代理对象:

        确定目标对象,即需要进行增强的对象。AOP 框架会创建一个代理对象来包含目标对象和切面。应用程序中会通过代理对象来调用目标对象的方法。

        3)选择连接点:

        在应用程序执行过程中,AOP框架根据切入点的定义选择适当的连接点。连接点是指程序执行过程中可以被增强的具体事件。

        4)执行切面逻辑:

        对于选择的连接点,AOP 框架会在该连接点上执行相应的增强逻辑,即通知。根据通知的类型,在连接点执行前、执行后或执行前后都可能执行切面逻辑。

        5)织入切面:

        织入是将切面与应用程序的目标对象结合起来创建代理对象的过程。AOP 框架会动态地将切面织入到目标对象的方法调用中,从而实现横切关注点的功能。织入可以发生在编译时、加载时、运行时或动态切入时。

        6)执行增强后的程序:

        当应用程序使用代理对象调用目标对象的方法时,会触发代理对象的增强逻辑。代理对象会在适当的连接点上执行切面逻辑,从而实现对应用程序的增强功能。

        简单来说,当程序运行时,执行到了与切入点匹配适合的连接点,也就是匹配到对应的方法时,那么就会由代理对象替代原始的方法,代理对象的方法包含了切面方法和原始的方法,也就是包含了通知与原始方法的代码,到最后,当程序执行原始方法的方法名的时候,不会继续往下执行原始方法里面的内容了,会执行代理对象中的方法。

        3.0 AOP 通知类型

        1)@Around:环绕通知,此注解标注的通知方法在目标方法前、后都被执行。

        2)@Before:前置通知,此注解标注的通知方法在目标方法前被执行。

        3)@After:后置通知,此注解标注的通知方法在目标方法后被执行,无论是否有异常都会执行。

        4)@AfterReturning:返回后通知,此注解标注的通知的方法在目标方法后被执行,有异常不会被执行,也就是说,当目标方法出现异常时,那么该通知方法就不会执行。

        5)@AfterThrowing:异常后通知,此注解标注的通知方法发生异常后执行。

需要注意的是:

        @Around 环绕通知需要自己调用 ProceedingJoinPoint.proceed() 来让原始方法执行,其他通知不需要考虑目标方法执行。@Around 环绕通知方法的返回值,必须指定为 Object ,来接收原始方法的返回值。

        4.0 AOP 通知顺序

        如果有多个通知类型都绑定在同一个连接点上,其执行顺序可能会有所不同。因此,在配置 AOP 时,需要谨慎考虑通知的顺序以保证业务逻辑的正确执行。

        也就是说,当连接点匹配到多个通知类型时,是按照什么顺序执行的呢?

        1)默认按照切面类的类名字母排序

        2)用 @Order(数字) 注解加在切面类上来控制顺序

        4.1 默认按照切面类的类名字母排序

        目标方法前的通知方法:字母排名靠前的先执行。

        目标方法后的通知方法:字母排名靠前的后执行。

代码演示:

demo1 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;@Component
@Aspect
public class demo1 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo1 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo1 切面类");return result;}
}

demo2 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;@Component
@Aspect
public class demo2 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo2 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo2 切面类");return result;}}

demo3 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;@Component
@Aspect
public class demo3 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo3 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo3 切面类");return  result;}
}

运行结果:

        4.2 用 @Order(数字) 注解加在切面类上来控制顺序

        目标方法前的通知方法:数字小的先执行。

        目标方法后的通知方法:数字小的后执行。

代码演示:

demo1 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;@Component
@Aspect
@Order(3)
public class demo1 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo1 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo1 切面类");return result;}
}

demo2 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;@Component
@Aspect
@Order(2)
public class demo2 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo2 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo2 切面类");return result;}
}

demo3 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;@Component
@Aspect
@Order(1)
public class demo3 {@Around("execution(* org.example.controller.DeptController.getList())")public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("正在执行 demo3 切面类");Object result = joinPoint.proceed();System.out.println("正在执行 demo3 切面类");return  result;}
}

运行结果:

        5.0 AOP 切入点表达式

        切入点表达式是描述切入点方法的一种表达式,用来筛选连接点也就是选择目标方法,主要用来决定项目中的哪些方法需要加入通知。

常见的形式:

        1)execution():根据方法的签名来匹配。

        2)@annotation():根据注解匹配。

补充:什么是方法签名?

        方法签名是一个方法在源代码中的表示,它由方法的名称、返回类型、参数列表以及可能的抛出异常列表组成。而权限修饰符是不属于方法签名的一部分。

        5.1 使用 execution() 创建切入点表达式

        1)根据方法的签名来匹配连接点。

        execution 主要根据方法的返回值、包名、类名、方法名、方法参数等信息来匹配,语法为:

execution(访问修饰符  返回值  包名.类名.方法名(方法参数) throws 异常)

        其中访问修饰符、包名.类名、throws 异常这些部分代码是可以省略的。需要注意的是 throws 异常是方法上声明抛出的异常,不是实际抛出的异常。

        2)可以使用通配符描述切入点:

        *:单个独立的任意符号,可以通配任意返回值、包名、类名、方法名、任意类型的一个参数,也可以通配包、类、方法名的一部分。

        ..:多个连续的任意符号,可以通配任意层级的包,或任意类型、任意个数的参数。

举个例子:

        execution(* org.example.controller.DeptController.getList()) 切入点为:在任意返回值类型下的 org.example.controller 包下的 DeptController 类下的 getList 没有参数的方法。

        还可以根据业务需要可以使用 && 、 ||、 ! 来组合比较复杂的切入点表达式。 

        5.2 使用 @annotation 创建切入点表达式

        用于匹配标识有特定注解的方法。

        1)首先创建一个注解

代码演示:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {}

        在注解上加上两个元注解 @Retention 用来表示该注解什么时候生效,会被保留到运行时,可以通过反射机制在运行时获取注解信息。@Target 指定了注解可以应用的目标类型。

        2)接着手动给连接点也就是目标方法上加上自定义的注解,最后在 AOP 通知类型的注解属性中添加自定义注解的全类名。

代码演示:

    @Around("@annotation(org.example.Anto.Log)")public Object log(ProceedingJoinPoint proceedingJoinPoint){//目标方法执行之前,需要执行的代码proceedingJoinPoint.proceed();//目标方法执行之后,需要执行的代码
}

        6.0 AOP 连接点

        连接点简单来说就是 AOP 所控制的方法。

        在 Spring 中使用 JoinPoint 抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等。

        对于 @Around 通知,获取连接点信息只能使用 ProceedingJoinPoint 。

        对于其他四种通知,获取连接点信息只能使用 JoinPoint,它是 ProceedingJoinPoint 的父类型。

代码演示:

//获取目标类的类名
String className = proceedingJoinPoint.getTarget().getClass().getName();//获取目标方法名
String methodName = proceedingJoinPoint.getSignature().getName();//获取目标方法的方法参数
Object[] args = proceedingJoinPoint.getArgs();//获得目标方法的返回值
Object result = proceedingJoinPoint.proceed();

        7.0 AOP 案例 - 记录操作日志

         实现将每一次操作的操作信息记录到数据库中。

实现思路:

        先创建数据库:

        实现类:

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import java.time.LocalDateTime;@Data
@AllArgsConstructor
@NoArgsConstructor
public class OperateLog {private Integer id;private  Integer operateUser;private LocalDateTime operateTime;private String className;private String methodName;private String methodParams;private String returnValue;private Long costTime;
}

        AOPMapper 接口:

import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
import org.example.Pojo.OperateLog;@Mapper
public interface AOPMapper {@Insert("insert into operate_log(operate_time,class_name,method_name,method_params,return_value,cost_time) " +"values (#{operateTime},#{className},#{methodName},#{methodParams},#{returnValue},#{costTime})")public void log(OperateLog operateLog);
}

        定义 AOP 切面类:

import com.alibaba.fastjson.JSONObject;
import io.jsonwebtoken.Claims;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.example.Pojo.OperateLog;
import org.example.Utilities.JWT;
import org.example.mapper.AOPMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;import javax.servlet.http.HttpServletRequest;
import java.time.LocalDateTime;
import java.util.Arrays;@Component
@Aspect
public class AOPLog {@AutowiredHttpServletRequest request;@AutowiredAOPMapper aopMapper;@Around("@annotation(org.example.Anto.Log)")public Object log(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {String jwt = request.getHeader("token");Claims claims = JWT.parse(jwt);LocalDateTime operateTime = LocalDateTime.now();//获取目标类的类名String className = proceedingJoinPoint.getTarget().getClass().getName();//获取目标方法名String methodName = proceedingJoinPoint.getSignature().getName();//获取目标方法的方法参数Object[] args = proceedingJoinPoint.getArgs();String methodParams = Arrays.toString(args);System.out.println("方法执行之前");long start = System.currentTimeMillis();//获得目标方法的返回值Object result = proceedingJoinPoint.proceed();long end = System.currentTimeMillis();String returnValue = JSONObject.toJSONString(result);Long costTime = end - start;OperateLog operateLog = new OperateLog(null,null,operateTime,className,methodName,methodParams,returnValue,costTime);aopMapper.log(operateLog);System.out.println(operateLog);System.out.println("方法执行之后");return result;}
}

        自定义的注解:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {}

         接着将 @Log 注解加到需要进行操作时记录的方法上即可。

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

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

相关文章

Java 还能不能继续搞了?

金三银四招聘季已落幕&#xff0c;虽说行情不是很乐观&#xff0c;但真正的强者从不抱怨。 在此期间&#xff0c;我收到众多小伙伴的宝贵反馈&#xff0c;整理出132道面试题&#xff0c;从基础到高级&#xff0c;有八股文&#xff0c;也有对某个知识点的深度解析。包括以下几部…

CC++内存管理【new和delete操作符的详细分析】【常见面试题】

C/C内存管理 1.C/C内存分布 我们先来看一段代码&#xff0c;来了解一下C/C中的数据内存分布。 # include <stdlib.h>int globalVar 1; static int staticGlobalVar 1; // 比globalVar还要先销毁,同一个文件下后定义的先析构 // 全局变量存在 数据段&#xff08;静态…

《尚庭公寓》项目部署之Docker + Nginx

docker rmi nginx docker pull nginx docker rm -f nginx #先创建一个简易的nginx容器&#xff08;后面会删&#xff09;&#xff0c;然后通过 docker cp命令把容器里面的nginx配置反向拷贝到宿主主机上。 docker run --name nginx -p 80:80 -d nginx# 将容器nginx.conf文件复…

【Linux】ip命令详解

Linux网络排查 目录 一、ip命令介绍 1.1 ip命令简介 1.2 ip命令的由来 二、ip命令使用帮助 2.1 ip命令的help帮助信息 2.2 ip命令对象介绍 2.3 ip命令选项介绍 三、查看网络信息 3.1 显示当前网络接口信息 3.2 显示网络设备运行状态 3.3 显示详细设备信息 3.4 查看…

“新夏入汉城,昂首度良辰”—Anzo Capital燃动武汉交易技术峰会

“2024年武汉交易技术峰会”在中国湖北武汉举办。Anzo Capital昂首资本作为2024年交易峰会的独家赞助商出席本次活动&#xff0c;Anzo Capital燃动现场&#xff0c;尽展昂扬奋进之姿。 活动现场&#xff0c; Anzo Capital昂首资本凭借无与伦比的交易环境、专业优质的服务、丰富…

【Python Cookbook】S01E21 文本模式的匹配和查找 match()、search()、findall() 以及 捕获组和 + 的含义

目录 问题解决方案讨论 问题 本文讨论一些按照特定的文本模式进行的查找和匹配。 解决方案 如果想要匹配的只是简单文字&#xff0c;通常我们使用一些内置的基本字符串方法即可&#xff0c;如&#xff1a;str.find()&#xff0c;str.startwith()&#xff0c;str.endswith() …

css 前端面试题学习思维导图学习笔记

嗨&#xff0c;我是小路。今天主要和大家分享的主题是“前端面试题学习笔记”。 一、面试题内容 1.link 和 import的区别 注意&#xff1a;在前端开发中&#xff0c;主要使用的是link,用import的比较少&#xff0c;只有在vue中会用到后者&#xff0c;尤其是加载顺序…

供应链管理怎么做?一文搞懂供应链数字化转型方案

供应链管理不仅关系到产品从原材料到成品&#xff0c;再到最终用户的整个流程&#xff0c;更是企业运营效率、成本控制和市场响应速度的重要体现。然而&#xff0c;在现代商业环境下&#xff0c;传统的供应链管理方式往往存在库存管理困难、协作效率低、结构不灵活等问题&#…

【机器学习】AI大模型的探索—分析ChatGPT及其工作原理

&#x1f4dd;个人主页&#xff1a;哈__ 期待您的关注 目录 &#x1f4da;介绍ChatGPT 1.1 什么是ChatGPT 1.2 ChatGPT的应用场景 &#x1f4a1;基础概念 1. 人工智能和机器学习 1.1 人工智能&#xff08;AI&#xff09;简介 1.2 机器学习&#xff08;ML&#xff09;简…

【语音告警】Zabbix语音播报-报警媒介部分配置-语音报警灯|声光报警器|网络信号灯

阅读说明 本文为博灵语音通知终端与Zabbix报警媒介的配置&#xff0c;对接完成后可以实现Zabbix的声光语音告警&#xff0c;播报效果可以参考 Modbus-博灵语音通知终端与PLC联动告警介绍 对接前需配置好通知终端的IP地址&#xff0c;设备参数参见 其他完整的Zabbix语音播报报…

AMPL下载安装于基本使用

1 注册安装 先去AMPL官网用邮箱注册 注册后按照提示下载社区版&#xff0c;社区版中&#xff0c;各种求解器都有30天的免费试用权限。下载安装包的时候&#xff0c;如果觉得太慢&#xff0c;可以将下载链接复制到迅雷&#xff0c;迅雷下载起来快很多。 2 新建文件并运行 安…

史上最全,呕心沥血总结oracle推进SCN方法(五)

作者介绍&#xff1a;老苏&#xff0c;10余年DBA工作运维经验&#xff0c;擅长Oracle、MySQL、PG数据库运维&#xff08;如安装迁移&#xff0c;性能优化、故障应急处理等&#xff09; 公众号&#xff1a;老苏畅谈运维 欢迎关注本人公众号&#xff0c;更多精彩与您分享。前面介…

【大事件】docker可能无法使用了

今天本想继续学习docker的命令&#xff0c;突然发现官方网站的文档页面打不开了。 难道是被墙了&#xff1f; 我用同事的翻了一下&#xff0c;能进&#xff0c;果然&#xff01; 正好手头的工作告一段落&#xff0c;将代码上传&#xff0c;然后通过jenkins将服务器自动部署到…

基于pytorch的车牌识别

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 一、导入数据 from torchvision.transforms import transforms from torch.utils.data import DataLoader from torchvision import datase…

RocketMQ可视化界面安装

RocketMQ可视化界面安装 **起因&#xff1a;**访问rocketmq-externals项目的git地址&#xff0c;下载了源码&#xff0c;在目录中并没有找到rocketmq-console文件夹。 git下面文档提示rocketMQ的仪表板转移到了新的项目中&#xff0c;点击仪表板到新项目地址&#xff1b; 下载…

计算机视觉与模式识别实验2-2 SIFT特征提取与匹配

文章目录 &#x1f9e1;&#x1f9e1;实验流程&#x1f9e1;&#x1f9e1;SIFT算法原理总结&#xff1a;实现SIFT特征检测和匹配通过RANSAC 实现图片拼接更换其他图片再次测试效果&#xff08;依次进行SIFT特征提取、RANSAC 拼接&#xff09; &#x1f9e1;&#x1f9e1;全部代…

ROG CETRA II 降临2代RGB版 使用体验!

现在Type-C接口的设备越来越多&#xff0c;不仅是台式机开始普及&#xff0c;像NUC、笔记本、Switch、安卓手机等也都是Type-C接口了&#xff0c;所以游戏耳机方面也开始迭代。Type-C还有一个好处就是供电足以撑起降噪处理和RGB灯效&#xff0c;你懂的。今天跟大家分享的就是RO…

CentOS 7~9 救援模式恢复root密码实战指南

在管理Linux服务器时&#xff0c;忘记root密码是一件棘手的事情&#xff0c;但幸运的是&#xff0c;CentOS提供了救援模式来帮助我们重置root密码。本文将详细介绍如何通过GRUB引导菜单进入紧急模式&#xff08;或称为救援模式&#xff09;&#xff0c;进而恢复root用户的密码。…

【ArcGISProSDK】 读取多面体信息并导出XML

结果展示 代码 using ArcGIS.Core.CIM; using ArcGIS.Core.Data; using ArcGIS.Core.Data.DDL; using ArcGIS.Core.Geometry; using ArcGIS.Core.Internal.CIM; using ArcGIS.Desktop.Catalog; using ArcGIS.Desktop.Core; using ArcGIS.Desktop.Editing; using ArcGIS.Deskto…

StableDiffusion简单使用教程

以下是一个简单的Stable Diffusion使用教程 一&#xff1a;准备工作 1. 安装所需软件&#xff1a;下载并安装 Stable Diffusion 相关程序。 2. 配置硬件&#xff1a;建议具备一定性能的显卡&#xff0c;以确保流畅运行。 二、启动软件 1. 打开 Stable Diffusion 应用程序。…