Vor kurzem, als ich Lebensläufe überarbeitete und Probeinterviews für viele Leute führte, gaben mir einige Freunde Feedback zu Fragen zu Spring AOP-Interviews, und ich werde sie heute stellen.
Das Mächtigste an Spring sind zu Beginn die beiden Kernfunktionen von IOC/AOP. Heute lernen wir die allgemeinen Anmerkungen und die Ausführungssequenz von Spring AOP.
Kernpunkte des Spring-Interviews:
IOC, AOP, Bean-Injektion, Bean-Lebenszyklus, Bean-Zirkelabhängigkeit
Lassen Sie uns zunächst einige häufig verwendete Anmerkungen in Spring Aop überprüfen:
@Before
前置通知:目标方法之前执行@After
后置通知:目标方法之后执行(始终执行)@AfterReturning
返回之后通知:执行方法结束之前执行(异常不执行)@AfterThrowing
异常通知:出香异常后执行@Around
@AfterThrowing
Ausnahmebenachrichtigung: Wird nach der Weihrauchausnahme ausgeführt
@Around
Around-Benachrichtigung: Ausführung der Zielmethode umgeben🎜🎜🎜🎜🎜🎜FAQ🎜🎜🎜 🎜🎜🎜1. Das müssen Sie Kennen Sie Spring, sprechen wir über die Reihenfolge aller Benachrichtigungen in Aop. Wie wirkt sich Spring Boot oder Spring Boot 2 auf die Ausführungsreihenfolge von Aop aus? 🎜🎜2. Erzählen Sie mir von den Fallstricken, auf die Sie bei AOP gestoßen sind? 🎜
Beispielcode
Lassen Sie uns schnell ein Spring Aop-Demoprogramm erstellen, um einige Details von Spring Aop zu besprechen.
Konfigurationsdatei
Um mir die direkte Verwendung von Spring-Boot für die schnelle Projektkonstruktion zu erleichtern, können Sie die Spring-Boot-Projekt-Schnellerstellungsfunktion von idea verwenden oder zu start.spring.io</codestil></p>
上面去快速创建spring-boot 应用。因为本人经常手动去网上贴一些依赖导致,依赖冲突服务启动失败等一些问题。
plugins { id 'org.springframework.boot' version '2.6.3' id 'io.spring.dependency-management' version '1.0.11.RELEASE' id 'java' } group 'io.zhengsh' version '1.0-SNAPSHOT' repositories { mavenCentral() maven { url 'https://repo.spring.io/milestone' } maven { url 'https://repo.spring.io/snapshot' } } dependencies { # 其实这里也可以不增加 web 配置,为了试验简单,大家请忽略 implementation 'org.springframework.boot:spring-boot-starter-web' implementation 'org.springframework.boot:spring-boot-starter-actuator' implementation 'org.springframework.boot:spring-boot-starter-aop' testImplementation 'org.springframework.boot:spring-boot-starter-test' } tasks.named('test') { useJUnitPlatform() }
首先我们需要定义一个接口。我们这里可以再来回顾一下 JDK 的默认代理实现的选择:
这块的逻辑在 DefaultAopProxyFactory
Da ich einige Abhängigkeiten häufig manuell online poste, treten Abhängigkeitskonflikte auf einige Probleme, wie z. B. ein Dienststartfehler.
public interface CalcService { public int div(int x, int y); }
Zuerst müssen wir eine Schnittstelle definieren. Hier können wir die Auswahl der Standard-Proxy-Implementierung von JDK überprüfen:
@Aspect @Component public class MyAspect { @Pointcut("execution(* io.zhengsh.spring.service.impl..*.*(..))") public void divPointCut() { } @Before("divPointCut()") public void beforeNotify() { System.out.println("----===>> @Before 我是前置通知"); } @After("divPointCut") public void afterNotify() { System.out.println("----===>> @After 我是后置通知"); } @AfterReturning("divPointCut") public void afterReturningNotify() { System.out.println("----===>> @AfterReturning 我是前置通知"); } @AfterThrowing("divPointCut") public void afterThrowingNotify() { System.out.println("----===>> @AfterThrowing 我是异常通知"); } @Around("divPointCut") public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { Object retVal; System.out.println("----===>> @Around 环绕通知之前 AAA"); retVal = proceedingJoinPoint.proceed(); System.out.println("----===>> @Around 环绕通知之后 BBB"); return retVal; } }🎜🎜🎜aop Interceptor🎜🎜🎜🎜Deklarieren Sie einen Interceptor und wir müssen @Aspect und @Component zum aktuellen Objekt hinzufügen. Der Autor ist bisher nur auf eine solche Grube getreten und hat nur eines hinzugefügt. 🎜
其实这块我刚开始也不是很理解,但是我看了 Aspect 注解的定义我就清楚了
这里面根本就没有 Bean 的定义。所以我们还是乖乖的加上两个注解。
还有就是如果当测试的时候需要开启Aop 的支持为配置类上增加@EnableAspectJAutoProxy
注解。
其实 Aop 使用就三个步骤:
@Aspect @Component public class MyAspect { @Pointcut("execution(* io.zhengsh.spring.service.impl..*.*(..))") public void divPointCut() { } @Before("divPointCut()") public void beforeNotify() { System.out.println("----===>> @Before 我是前置通知"); } @After("divPointCut") public void afterNotify() { System.out.println("----===>> @After 我是后置通知"); } @AfterReturning("divPointCut") public void afterReturningNotify() { System.out.println("----===>> @AfterReturning 我是前置通知"); } @AfterThrowing("divPointCut") public void afterThrowingNotify() { System.out.println("----===>> @AfterThrowing 我是异常通知"); } @Around("divPointCut") public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { Object retVal; System.out.println("----===>> @Around 环绕通知之前 AAA"); retVal = proceedingJoinPoint.proceed(); System.out.println("----===>> @Around 环绕通知之后 BBB"); return retVal; } }
其实我这个测试类,虽然用了 @Test 注解,但是我这个类更加像一个 main 方法把:如下所示:
结果记录:spring 4.x, spring-boot 1.5.9
无法现在依赖,所以无法试验
我直接说一下结论:Spring 4 中环绕通知是在最里面执行的
结果记录:spring 版本5.3.15 springboot 版本2.6.3
多个切面的情况下,可以通过@Order指定先后顺序,数字越小,优先级越高。如下图所示:
下面一种场景会导致 aop 代理失效,因为我们在执行 a 方法的时候其实本质是执行 AServer#a
的方法拦截器(MethodInterceptor
)链, 当我们在 a 方法内直接执行b(), 其实本质就相当于 this.b() , 这个时候由执行 a方法是调用到 a 的原始对象相当于是 this 调用,那么会导致 b() 方法的代理失效。这个问题也是我们开发者在开发过程中最常遇到的一个问题。
@Service public class AService { public void a() { System.out.println("...... a"); b(); } public void b() { System.out.println("...... b"); } }
Das obige ist der detaillierte Inhalt vonInterviewer: Allgemeine Anmerkungen und Ausführungssequenz von Spring Aop. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!