Spring AOP官方文档学习笔记(四)之Spring AOP的其他知识点

1.选择哪种AOP

(1) 使用Spring AOP比使用完整版的AspectJ更方便简单,因为不需要在开发和构建过程中引入AspectJ编译器以及织入器,如果我们只希望通知能够在Spring Bean上执行,那么选用Spring AOP就可以了,如果我们希望通知能够在不由Spring所管理的对象上执行,那么就需要使用AspectJ,如果我们希望为除方法以外的连接点(比如成员变量)提供通知,那么也需要使用AspectJ

2.Spring AOP的代理机制

(1) Spring AOP使用Jdk动态代理或Cglib动态代理来为目标对象创建代理对象,Jdk动态代理由Jdk提供,而Cglib动态代理则是由一个开源类库提供,如果要代理的目标对象至少实现了一个接口,那么就会使用Jdk动态代理,否则如果目标对象没有实现任何接口,那么就会使用Cglib动态代理,创建一个Cglib代理对象

(2) Spring默认使用Jdk动态代理,但我们可以强制让Spring始终使用Cglib动态代理,但需注意,使用Cglib动态代理,无法对final修饰的方法织入通知,因为这些方法不能在子类中被重写,具体开启Cglib动态代理的方式如下

<!-- 方式一:在使用基于xml的配置时,设置<aop:config/>标签中的proxy-target-class属性为true -->
<aop:config proxy-target-class="true">
        <!-- ... -->
</aop:config>

<!-- 方式二:在混合使用基于xml和注解的配置时,设置<aop:aspectj-autoproxy/>标签中的proxy-target-class属性为true -->
<aop:aspectj-autoproxy proxy-target-class="true"/>

<!-- 方式三:在使用基于注解的配置时,设置@EnableAspectJAutoProxy注解中的proxyTargetClass属性为true -->
@EnableAspectJAutoProxy(proxyTargetClass = true)

(3) 当一个bean被代理后,我们从容器中获取到这个bean,并对其使用 .getClass().getName() 方法来输出它的类名称,可见如 cn.example.spring.boke.ExampleA$$EnhancerBySpringCGLIB$$ff6c22d2或com.sun.proxy.$Proxy18 这样的输出,而当我们关闭掉AOP后,得到的通常是形如 cn.example.spring.boke.ExampleA 这样的输出,这其实是因为我们从容器中获取的是该bean被增强过后的代理对象,而非它原始的目标对象,因而,对这个bean的方法调用就是对代理对象的方法调用,然后由代理对象委托调用原始对象上相关的方法以及该方法相关的拦截器(advice),如下

Spring AOP官方文档学习笔记(四)之Spring AOP的其他知识点

(4) 在目标对象中,使用this指针进行自调用不会触发通知的执行

//一个普通的bean,在它的a方法中使用this指针,自调用b方法
@Component
public class ExampleA{
    public void a() {
        System.out.println("a...");
        this.b();
    }

    public void b() {
        System.out.println("b...");
    }
}

//切面,切入ExampleA这个bean中的所有方法
@Component
@Aspect
public class Logger {
    @Before(value = "execution(* cn.example.spring.boke.ExampleA.*(..))")
    public void beforePrint() {
        System.out.println("beforePrint...");
    }
}

@Configuration
@EnableAspectJAutoProxy
@ComponentScan(basePackages = "cn.example.spring.boke")
public class Config { }

//获取ExampleA,调用a方法,打印结果
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(Config.class);
ExampleA exampleA = ctx.getBean(ExampleA.class);
exampleA.a();

//可见,Spring对a方法进行了织入,而b方法却没有,原因就是因为这里的this指向的是目标对象,一个普通的bean ExampleA,而非它的代理对象,自然而然无法进行织入了,因此关键的目标就是如何获取到代理对象
beforePrint...
a...
b...


//想要获取代理对象,首先要先将@EnableAspectJAutoProxy注解中的exposeProxy属性设置为true
@EnableAspectJAutoProxy(exposeProxy = true)

//接着修改ExampleA中的a方法,在调用b方法时不再使用this指针,而是AopContext.currentProxy(),即获取当前对象的代理对象
public void a() {
    System.out.println("a...");
//        this.b();
    ((ExampleA) AopContext.currentProxy()).b();
}

//接着,再执行打印,可见此时通知已被正确执行
beforePrint...
a...
beforePrint...
b...

Spring不推荐使用如上的方法,因为这会使Spring AOP与我们的代码强耦合,具有侵入性,最好的方式是重构我们的代码,避免发生自调用,此外,Spring AOP会产生这种问题的原因是Spring AOP是基于代理实现的,而AspectJ框架就不存在这种自调用问题,因为它不是一个基于代理的AOP框架

3.基于java代码的AOP

public class ExampleA{
    public void a() {
        System.out.println("a...");
    }
}

//切面必须由@Aspect注解标注,否则容器会抛出异常
@Aspect
public class Logger {
    @Before(value = "execution(* cn.example.spring.boke.ExampleA.*(..))")
    public void beforePrint() {
        System.out.println("beforePrint...");
    }
}

//创建AOP工厂,生成代理对象
public static void main(String[] args) throws Exception {
    //1.使用AspectJProxyFactory工厂,用于生成目标对象的代理对象
    AspectJProxyFactory factory = new AspectJProxyFactory(new ExampleA());

    //2.添加一个切面,该切面必须由@Aspect注解标注
    factory.addAspect(Logger.class);

    //3.生成代理对象
    ExampleA proxy = factory.getProxy();
    proxy.a();
}

原文链接:https://www.cnblogs.com/shame11/p/17370702.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Spring AOP官方文档学习笔记(四)之Spring AOP的其他知识点 - Python技术站

(0)
上一篇 2023年5月10日
下一篇 2023年5月10日

相关文章

  • SpringMVC上传文件的两种方法

    在 Spring MVC 中,我们可以使用两种方法来上传文件,分别是使用 MultipartFile 类型的参数和使用 CommonsMultipartResolver 类。本文将详细讲解 Spring MVC 上传文件的两种方法,包括如何使用 MultipartFile 类型的参数和如何使用 CommonsMultipartResolver 类,并提供两个…

    Java 2023年5月18日
    00
  • 各类常见语言清除网页缓存方法汇总

    各类常见语言清除网页缓存方法汇总 在开发Web应用时,经常会涉及到网页缓存的问题。为了尽可能的保证用户体验,我们需要清除掉网页缓存,使得每次用户访问页面时都能获取最新的数据。本篇文章将针对以下常见语言,为大家汇总清除网页缓存的方法。 HTML 用户访问网站时,浏览器会自动缓存显示的网页,以便下次快速加载。清除浏览器的缓存可以通过以下方法: <meta …

    Java 2023年6月15日
    00
  • Java的Struts框架报错“ActionMappingNotFoundException”的原因与解决办法

    当使用Java的Struts框架时,可能会遇到“ActionMappingNotFoundException”错误。这个错误通常由以下原因之一起: 配置错误:如果配置文件中没有正确配置Action映射,则可能会出现此。在这种情况下,需要检查配置文件以解决此问题。 编码错误:如果代码中没有正确设置Action映射,则可能会出现此。在这种情况下,需要检查代码以解…

    Java 2023年5月5日
    00
  • Java中的ClassNotFoundException是什么?

    ClassNotFoundException是Java中的一种异常类型,表示虚拟机在试图加载类时无法找到指定的类。 当Java虚拟机无法找到某个类时,会抛出ClassNotFoundException异常。通常情况下,这种情况发生在以下几种情形中: 使用Class.forName()方法加载类时,指定的类不存在; 使用ClassLoader.loadClas…

    Java 2023年4月27日
    00
  • 在Java的Spring框架的程序中使用JDBC API操作数据库

    使用JDBC API操作数据库是Java程序开发的基本技能之一,而在Spring框架中使用JDBC则是非常常见的情况。下面是在Java的Spring框架中使用JDBC API操作数据库的完整攻略。 配置JDBC数据源 在Spring中,我们需要先配置一个JDBC数据源。数据源的配置通常是在Spring的XML配置文件中完成的。如下是一个典型的JDBC数据源配…

    Java 2023年5月20日
    00
  • 什么是并行收集器?

    下面我来详细讲解一下“什么是并行收集器?”的完整使用攻略。 并行收集器是什么? 并行收集器就是一种并行执行的垃圾收集器,它利用多个线程同时进行垃圾收集。它针对的是堆内存比较大的场景,因为在这种场景下,垃圾收集器需要进行很多的扫描和标记操作,使用多线程可以有效加快垃圾收集的速度。 如何使用并行收集器? 使用并行收集器很简单,只需要使用以下参数即可: -Xmx&…

    Java 2023年5月10日
    00
  • Java程序控制逻辑—流程控制

    关于“Java程序控制逻辑—流程控制”的完整攻略,我会从以下几个方面进行讲解: 流程控制的基本概念 条件语句 循环语句 例子说明 1. 流程控制的基本概念 在编写Java程序时,我们需要按照一定的逻辑来控制程序的执行顺序。流程控制就是指通过条件判断和循环来控制程序中语句的执行顺序,使程序按照我们设定的逻辑进行。 Java的流程控制主要有两种:条件语句和循环语…

    Java 2023年5月23日
    00
  • SpringMVC @RequestBody出现400 Bad Request的解决

    下面我为您详细讲解“SpringMVC @RequestBody出现400 Bad Request的解决”的完整攻略。 问题描述 在使用SpringMVC框架中,我们经常会用到 @RequestBody 注解来接收 HTTP 请求中的参数。但是,有时候我们会遇到使用 @RequestBody 得到 400 Bad Request 的错误响应码的情况。这是什么…

    Java 2023年5月26日
    00
合作推广
合作推广
分享本页
返回顶部