spring全家桶帮助java web开发者节省了不少开发量,提高了效率。可是由于屏蔽了不少细节,致使不少开发者只知其然,不知其因此然,本文就是分析下使用spring的一些注解,不可以自调用的问题。由于自己这类文章不少,因此有些地方不会详述,直接引用其余文章。html
@Cache
, @Async
,@Transaction
这三种原理上有什么区别吗首先须要澄清几个须要区分的名词 AOP
Spring AOP
AspectJ
java
Aspect-oriented programming,面向切面编程,一种解决问题的思想,将一些重复性的编码问题经过切面来实现。 不少人了解切面是经过Spring来了解的,因此会有种误解将SpringAOP和AOP划等号,其实否则。web
Spring AOP 算是一种简单的AOP的落地实现方式,它主要提供在Spring容器内的一种AOP实现方式,脱离了Spring就不work了。Spring AOP并非一套完整的AOP解决方案。spring
Spring的的众多组件都是这样,Spring-Session,Spring-jdbc,Spring-Cache等等,都能解决一部分通用的需求,可是会有不少限制, 想用深了,更灵活的实现功能,仍是要使用其余的专业组件/框架。编程
SpringAOP默认使用代理模式实现的,也就是JDK Proxy/CGLib。关于代理以及JDK Proxy和CGLib不在赘述了。 api
Spring AOP并非一套完整的AOP解决方案,AspectJ是的。AspectJ在编译器织入切面到目标类bash
上面介绍了SpringAop的实现,下面着重介绍解法。框架
这个原理没啥好解析的async
@Autowired
@Lazy
private AsyncMethod asyncMethod;
public void testAsync() {
System.out.println(Thread.currentThread().getName());
// 调用注入的bean
asyncMethod.testAsnc3();
}
@Async
public void testAsnc3() {
System.out.println(Thread.currentThread().getName());
System.out.println("async3");
}
复制代码
会有循环依赖的问题,使用@Lazy
解决测试
AopContext.currentProxy()
获取当前代理对象首先须要配置@EnableAspectJAutoProxy(exposeProxy = true)
,容许代码中获取proxy类
public void testAsync() {
System.out.println(Thread.currentThread().getName());
System.out.println("async1");
((AsyncMethod)AopContext.currentProxy()).testAsnc2();
}
@Async
public void testAsnc2() {
System.out.println(Thread.currentThread().getName());
System.out.println("async2");
}
复制代码
这个实现能够看下AopContext类,
// 经过ThreadLocal来实现的
private static final ThreadLocal<Object> currentProxy = new NamedThreadLocal<Object>("Current AOP proxy");
复制代码
而后就是Spring Aop自动设置代理,设置exposeProxy
属性的问题了。 有人写过了,就不写了
既然自调用的问题是因为SpringAOP由代理模式实现引发的,那就不使用代理模式不就解决了吗
@EnableAsync(mode = AdviceMode.ASPECTJ)
复制代码
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.8</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-instrument -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-instrument</artifactId>
<version>4.2.5.RELEASE</version>
</dependency>
复制代码
public void testAsync() {
System.out.println(Thread.currentThread().getName());
System.out.println("async1");
testAsnc2();
}
/**
* 测试ASPECTJ调用
*/
@Async
private void testAsnc2() {
System.out.println(Thread.currentThread().getName());
System.out.println("async2");
}
复制代码
-Dserver.port=1000 -javaagent:${classpath}\spring-instrument-4.2.5.RELEASE.jar
-javaagent:${classpath}\aspectjweaver-1.8.8.jar
复制代码
方法 | 限制 |
---|---|
自调用 | 代理模式的限制,好比只能做用于public ,非static的方法 |
AopContext.currentProxy() | 1. 代理模式的限制 2.ThreadLocal的限制,不能跨线程了 3.bean设置的限制,好比@Async代理建立方式不一样其余 |
AspectJ | 无限制,使用起来麻烦一点 |
关注公众号【方丈的寺院】,第一时间收到文章的更新,与方丈一块儿开始技术修行之路