Spring框架是Java开发中最流行的应用程序框架之一。它提供了广泛的功能,其中之一就是面向切面编程(AOP)。Spring AOP允许我们将关注点(例如日志记录、事务管理)模块化,并将它们应用到应用程序中的多个部分,而不是将它们散布在整个代码库中。
本文将深入探讨Spring AOP的底层原理,帮助更好地理解这一关键概念。
在Spring AOP中,切面是通知(Advice)和切点(Pointcut)的组合。通知定义了在何时以及何方式执行代码,而切点定义了何处执行这些代码。通常,切面用于跨越多个模块的关注点,例如日志记录或事务管理。
通知是切面的核心。它定义了在切点处执行的代码。
切点定义了通知应该被执行的位置。它使用表达式或规则匹配方法的名称和参数。例如,可以使用切点来匹配所有以“get”开头的方法。
Spring AOP依赖于代理模式来实现切面。将通知应用到目标对象之后,程序动态创建的通知对象,就称为代理。
代理类既可能是和原类具有相同接口的类,也可能是原类的子类,可以采用调用原类相同的方式调用代理类。也就是我们昨天说的如何判断是哪种动态代理方式的区别。
默认情况下,Spring AOP使用JDK动态代理,JDK动态代理是通过java.lang.reflect.Proxy 类实现的,可以调用Proxy类的newProxyInstance()方法创建代理对象。JDK动态代理可以实现无侵入式的代码扩展,并且可以在不修改源代码的情况下,通知/增强某些方法。
案例具体实现步骤如下:
<dependencies>
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-contextartifactId>
<version>5.3.10version>
dependency>
<dependency>
<groupId>org.aspectjgroupId>
<artifactId>aspectjweaverartifactId>
<version>1.9.7version>
dependency>
package com.steveDash.dao;
import com.steveDash.entity.User;
public interface UserDao {
public void saveUser(User user);
public void deleteUser(User user);
}
package com.steveDash.dao;
import com.steveDash.entity.User;
public class UserDaoImpl implements UserDao {
private UserDao userDao;
public void saveUser(User user) {
System.out.println("添加用户成功");
}
public void deleteUser(User user) {
System.out.println("删除用户成功");
}
}
package com.steveDash.dao;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
//定义的切面类:里面编写的是各种通知方法Advice(通知或者增强方法)
@Aspect
public class TestAspect {
@Pointcut("execution(public void com.steveDash.dao.UserDaoImpl.saveUser())")
public void pointcut(){}
@Before("pointcut()")
public void before(){
System.out.println("使用前检查是否满足条件?");
}
@Pointcut("execution(public void com.steveDash.dao.UserDaoImpl.deleteUser())")
public void pointcut1(){}
@AfterReturning("pointcut1()")
public void afterResult(){
System.out.println("调用结束后返回日志结果,进行记录");
}
}
package com.steveDash.jdkTest;
import com.steveDash.dao.UserDao;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class MyInvocationHandler implements InvocationHandler {
private UserDao target;
public MyInvocationHandler(UserDao target) {
this.target =target;
}
//若省略invoke()方法会报错,需要自行完成具体的代码体,方法框架可以一键生成,如下文
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 在方法调用前执行一些操作
System.out.println("Before invoking method: " + method.getName());
// 调用目标对象的方法
Object result = method.invoke(target, args);
// 在方法调用后执行一些操作
System.out.println("After invoking method: " + method.getName());
return result;
}
}
进行导包,选择第一个java.lang.reflect,跟我们前面说的一样
下面的Proxy也得进行导包,也是Java.lang.reflect
到这里就跟我上面的代码块的效果基本一致了,会发现还有报错的地方,这里的意思就是说缺少了invoke()方法的代码,让我们实现该方法,鼠标移至灰色处,点击Implement methods,选择invoke方法
仔细查看这一块的代码,并且可以尝试替换成TestAspect里面的Before和AfterReturing的方法,这里只是用简单的输出来模拟过程。
package com.steveDash.dao;
import com.steveDash.entity.User;
import com.steveDash.jdkTest.MyInvocationHandler;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
public class JDKTest {
public static void main(String[] args) {
// 创建实际的目标对象
UserDao userDao = new UserDaoImpl();
// 创建自定义的 InvocationHandler
InvocationHandler handler = new MyInvocationHandler(userDao);
// 使用 Proxy 创建代理对象
UserDao userDaoProxy = (UserDao) Proxy.newProxyInstance(
JDKTest.class.getClassLoader(),
new Class[]{UserDao.class},
handler
);
// 创建一个用户对象
User user = new User(1,"SteveDash");
// 通过代理对象调用方法
userDaoProxy.saveUser(user);
userDaoProxy.deleteUser(user);
}
}
整个过程就很明啦
JDK动态代理存在缺陷,它只能为接口创建代理对象,当需要为类创建代理对象时,就需要使用CGLib(Code Generation Library)动态代理,它采用底层的字节码技术,通过继承
的方式动态创建代理对象。
Spring的核心包已经集成了CGLib所需要的包,所以开发中不需要另外导入JAR包。
package com.steveDash.myDao;
public class UserDao {
public void saveUser() {
System.out.println("Saving user data...");
}
}
package com.steveDash.myDao;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class UserDaoCglibProxy implements MethodInterceptor {
private UserDao target;
public UserDaoCglibProxy(UserDao target) {
this.target = target;
}
public Object createProxy() {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
return enhancer.create();
}
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("Before method execution...");
Object result = proxy.invokeSuper(obj, args); // 调用目标对象的方法
System.out.println("After method execution...");
return result;
}
}
和上面一样进行导包,在爆红的地方
导入第一个带有cglib的
然后整体就是这个样子的
代码详解:
在这里**UserDaoCglibProxy
类实现了 MethodInterceptor
接口,它会在目标方法执行前后进行拦截**。
Enhancer
类用于创建代理对象,通过设置目标类的超类(setSuperclass
)和回调拦截器(setCallback
)来生成代理对象。
在拦截器的 intercept
方法中,我们可以在调用目标方法前后执行自定义的通知方法。就跟前面的AspectTest里面的通知一样。
从这里就可以看出,CGLib确实是动态代理成功并且拦截了我们的目标方法的调用
关于AOP的实现昨天我们已经编写过啦,这里就不再重复。三层架构设计模式MVC和AOP面向切面编程—SSM框架的学习与应用(Spring + Spring MVC + MyBatis)-Java EE企业级应用开发学习记录(第九天)_Stevedash的博客-CSDN博客
JDK动态代理和CGLib动态代理都是Java中用于实现代理模式的技术,它们允许我们创建代理对象来控制对其他对象的访问。
InvocationHandler
接口来创建代理对象,该接口包含一个方法 invoke
,在代理对象的方法被调用时,该方法会被执行。java.lang.reflect.Proxy
类来创建代理对象。net.sf.cglib.proxy.Enhancer
类来创建代理对象。MethodInterceptor
接口来实现代理逻辑,该接口包含一个方法 intercept
,在代理对象的方法被调用时,该方法会被执行。final
声明的类和方法。作者:Stevedash
发表于:2023年9月9日19点37分
注:本文内容基于个人学习理解,如有错误或疏漏,欢迎指正。感谢阅读!如果觉得有帮助,请点赞和分享。