MethodBeforeAdvice接口源码分析

master
linlei 2024-04-11 10:57:44 +08:00
parent 1bc64a0e52
commit 9f6a76f827
6 changed files with 185 additions and 0 deletions

View File

@ -30,6 +30,7 @@
<module>spring-aop-annotation-enableAspectJAutoProxy</module>
<module>spring-aop-annotation-enableLoadTimeWeaving</module>
<module>spring-aop-advice-methodInterceptor</module>
<module>spring-aop-advice-methodBeforeAdvice</module>
</modules>
<modelVersion>4.0.0</modelVersion>

View File

@ -0,0 +1,132 @@
## MethodBeforeAdvice
- [MethodBeforeAdvice](#MethodBeforeAdvice)
- [一、基本信息](#一基本信息)
- [二、基本描述](#二基本描述)
- [三、主要功能](#三主要功能)
- [四、接口源码](#四接口源码)
- [五、主要实现](#五主要实现)
- [六、最佳实践](#六最佳实践)
- [七、源码分析](#七源码分析)
- [八、常见问题](#八常见问题)
### 一、基本信息
✒️ **作者** - Lex 📝 **博客** - [掘金](https://juejin.cn/user/4251135018533068/posts) 📚 **源码地址** - [github](https://github.com/xuchengsheng/spring-reading)
### 二、基本描述
`MethodBeforeAdvice`接口是Spring AOP中的一个核心接口允许我们在目标方法执行之前插入自定义的逻辑例如参数验证、日志记录等从而实现面向切面编程的前置通知功能。
### 三、主要功能
1. **前置通知**
+ 允许开发者在目标方法执行之前执行额外的逻辑操作。
2. **横切关注点的分离**
+ 将与业务逻辑无关的横切关注点(如日志记录、性能监控、安全验证等)与核心业务逻辑分离开来,提高代码的模块化和可维护性。
3. **参数验证**
+ 在目标方法执行前对方法参数进行验证,确保参数的合法性。
4. **权限控制**
+ 在方法执行前进行权限检查,确保只有具有足够权限的用户能够执行该方法。
### 四、接口源码
`MethodBeforeAdvice`接口,用于在方法调用之前执行通知。通知方法`before`接收被调用的方法、方法参数以及方法调用的目标对象作为参数并可以抛出Throwable以中止方法调用。这样的通知无法阻止方法调用的继续进行除非它们抛出了Throwable。
```java
/**
* 在方法被调用之前调用的通知。这样的通知不能阻止方法调用的继续进行除非它们抛出了一个Throwable。
*
* @author Rod Johnson
* @see AfterReturningAdvice
* @see ThrowsAdvice
*/
public interface MethodBeforeAdvice extends BeforeAdvice {
/**
* 在给定方法被调用之前的回调。
* @param method 被调用的方法
* @param args 方法的参数
* @param target 方法调用的目标对象。可能为 {@code null}。
* @throws Throwable 如果此对象希望中止调用。任何抛出的异常如果方法签名允许,将返回给调用者。否则异常将作为运行时异常进行包装。
*/
void before(Method method, Object[] args, @Nullable Object target) throws Throwable;
}
```
### 五、主要实现
1. **AspectJMethodBeforeAdvice**
- 实现了前置通知,使用 AspectJ 风格定义的通知,用于在目标方法执行前执行额外的逻辑。
### 六、最佳实践
使用`MethodBeforeAdvice`接口。首先,通过创建代理工厂和目标对象,然后创建自定义的前置通知`MyMethodBeforeAdvice`,将其添加到代理工厂中。接着,通过代理工厂获取代理对象,并调用代理对象的方法。在方法调用之前,前置通知会被触发执行,执行自定义的逻辑。
```java
public class MethodBeforeAdviceDemo {
public static void main(String[] args) {
// 创建代理工厂&创建目标对象
ProxyFactory proxyFactory = new ProxyFactory(new MyService());
// 创建通知
proxyFactory.addAdvice(new MyMethodBeforeAdvice());
// 获取代理对象
MyService proxy = (MyService) proxyFactory.getProxy();
// 调用代理对象的方法
proxy.doSomething();
}
}
```
`MyMethodBeforeAdvice`类实现了`MethodBeforeAdvice`接口,在其`before`方法中打印出目标方法被调用之前的信息包括方法名。这个类可以用作Spring AOP中的前置通知在目标方法执行之前执行一些额外的逻辑。
```java
public class MyMethodBeforeAdvice implements MethodBeforeAdvice {
@Override
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println("Before method " + method.getName() + " is called.");
}
}
```
`MyService` 类是一个简单的服务类,其中包含了一个名为 `doSomething()` 的方法。在上下文中,`MyService` 类被用作目标对象,即需要被拦截和增强的对象。
```java
public class MyService {
public String doSomething() {
System.out.println("Doing something...");
return "hello world";
}
}
```
运行结果,调用目标方法`doSomething`之前,`MyMethodBeforeAdvice`中的前置通知被成功触发,并打印了相应的信息。
```java
Before method doSomething is called.
Doing something...
```
### 七、源码分析
暂无
### 八、常见问题
1. **前置通知的执行顺序问题**
+ 当一个类中有多个前置通知时它们的执行顺序是怎样的这可能涉及到AOP代理链中各个通知的调用顺序问题。
2. **目标方法参数获取**
+ 如何在前置通知中获取目标方法的参数?`before`方法的参数`args`提供了目标方法的参数数组,但如何准确地获取和处理这些参数可能需要进一步的了解。

View File

@ -0,0 +1,14 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<groupId>com.xcs.spring</groupId>
<artifactId>spring-aop</artifactId>
<version>0.0.1-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>spring-aop-advice-methodBeforeAdvice</artifactId>
</project>

View File

@ -0,0 +1,17 @@
package com.xcs.spring;
import org.springframework.aop.framework.ProxyFactory;
public class MethodBeforeAdviceDemo {
public static void main(String[] args) {
// 创建代理工厂&创建目标对象
ProxyFactory proxyFactory = new ProxyFactory(new MyService());
// 创建通知
proxyFactory.addAdvice(new MyMethodBeforeAdvice());
// 获取代理对象
MyService proxy = (MyService) proxyFactory.getProxy();
// 调用代理对象的方法
proxy.doSomething();
}
}

View File

@ -0,0 +1,12 @@
package com.xcs.spring;
import org.springframework.aop.MethodBeforeAdvice;
import java.lang.reflect.Method;
public class MyMethodBeforeAdvice implements MethodBeforeAdvice {
@Override
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println("Before method " + method.getName() + " is called.");
}
}

View File

@ -0,0 +1,9 @@
package com.xcs.spring;
public class MyService {
public String doSomething() {
System.out.println("Doing something...");
return "hello world";
}
}