### spring aop嵌套调用问题
在使用 Spring AOP 的时候,我们从 IOC 容器中获取的 Service Bean 对象其实都是代理对象,而不是那些 Service Bean 对象本身,也就是说获取的并不是被代理对象或代理目标。当我在自己的 Service 类中使用 this 关键字嵌套调用同类中的其他方法时,由于 this 关键字引用的并不是该 Service Bean 对象的代理对象,而是其本身,故 Spring AOP 是不能拦截到这些被嵌套调用的方法的
当同一个类中对两个方法增强后,一个方法调用另外一个方法时,产生了嵌套调用,其中一个切面就会失效
具体代码:
~~~
public void hello() {
System.out.println("Hello,IOC");
goodbye();
}
public void goodbye() {
System.out.println("Goodbye");
}
@Pointcut("execution(public * io.github.dunwu.spring.core.aop.example.IOCServiceImpl.hello(..))")
public void testAOP1(){
}
@Around("testAOP1()")
public Object around(ProceedingJoinPoint p){
System.out.println("around before testAOP1...");
Object o = null;
try {
o = p.proceed();
} catch (Throwable e) {
e.printStackTrace();
}
System.out.println("around after testAOP1...");
return o;
}
@Pointcut("execution(public * io.github.dunwu.spring.core.aop.example.IOCServiceImpl.goodbye(..))")
public void testAOP2(){
}
@Around("testAOP2()")
public void around(ProceedingJoinPoint p){
System.out.println("around before testAOP2...");
Object o = null;
try {
p.proceed();
} catch (Throwable e) {
e.printStackTrace();
}
System.out.println("around after testAOP2...");
return o;
}
~~~
调用hello方法时,会发生什么呢?
只有hello方法被增强了,goodbye方法直接被调用,并没有被增强。输出如下:
~~~undefined
around before testAOP1...
Hello,IOC
Goodbye
around after testAOP1...
~~~
** 为什么goodbye方法没有被增强 **
这是因为hello方法被增强之后,invoke hello方法时,调用goodbye其实是调用this.goodbye方法,也就是原始方法而并不是被增强的方法
### 原理
```
public interface ICustomerService {
public void doSomething1();
public void doSomething2();
}
public class CustomerServiceImpl implements ICustomerService {
public void doSomething1() {
System.out.println("Inside CustomerServiceImpl.doSomething1()");
doSomething2();
}
public void doSomething2() {
System.out.println("Inside CustomerServiceImpl.doSomething2()");
}
}
public class CustomerServiceProxy implements ICustomerService {
private ICustomerService customerService;
public void setCustomerService(ICustomerService customerService) {
this.customerService = customerService;
}
public void doSomething1() {
doBefore();
customerService.doSomething1();
doAfter();
}
public void doSomething2() {
doBefore();
customerService.doSomething2();
doAfter();
}
private void doBefore() {
// 例如,可以在此处开启事务
System.out.println("do some important things before...");
}
private void doAfter() {
// 例如,可以在此处提交或回滚事务、释放资源等等
System.out.println("do some important things after...");
}
}
public class TestProxy {
public static void main(String[] args) {
// 创建代理目标对象。对于Spring来说,这一工作
// 是由Spring DI容器完成的。
ICustomerService serviceProxyTarget = new CustomerServiceImpl();
// 创建代理对象。对于Spring来说,这一工作
// 也是由Spring DI容器完成的。
CustomerServiceProxy serviceProxy = new CustomerServiceProxy();
serviceProxy.setCustomerService(serviceProxyTarget);
ICustomerService serviceBean = (ICustomerService) serviceProxy;
// 调用业务逻辑操作
serviceBean.doSomething1();
}
}
```
现在以调试方式运行这个应用,你会发现在 doSomething1() 中调用 doSomething2() 方法的时候并未去执行CustomerServiceProxy 类的 doBefore()、doAfter() 方法。再来看看这句关键代码:doSomething2(); 把它隐含的意思也表达出来吧:this.doSomething2(); 在CustomerServiceImpl类中this关键字表示的是当前这个CustomerServiceImpl类的实例。那程序当然就会去执行 CustomerServiceImpl 类中的 doSomething2() 方法了,而不会去执行 CustomerServiceProxy 类中的 doSomething2() 方法
### 解决方案
1. 将自身注入到自身
```
public class OneBean {
@Resource
private OneBean oneBean;
public OneBean() {
System.out.println("构造器OneBean加载..." + this);
}
public String say(String a) {
System.out.println("say a=" + a);
oneBean.say2(a);
return a + a;
}
public String say2(String a) {
System.out.println("say2 a=" + a);
return a + a;
}
```
2. 使用AopContext.currentProxy()来操作
```
public class OneBean {
public OneBean() {
System.out.println("构造器OneBean加载..." + this);
}
public String say(String a) {
System.out.println("say a=" + a);
((OneBean)AopContext.currentProxy()).say2(a) ;
return a + a;
}
public String say2(String a) {
System.out.println("say2 a=" + a);
return a + a;
}
}
```
- java演变
- JDK各个版本的新特性
- JDK1.5新特性
- JDK1.6新特性
- JDK1.7新特性
- JDK1.8新特性
- JAVA基础
- 面向对象特性
- 多态
- 方法重载
- 方法重写
- class
- 常量
- 访问修饰符
- 类加载路径
- java-equals
- 局部类
- java-hashCode
- Java类初始化顺序
- java-clone方法
- JAVA对象实例化的方法
- 基础部分
- JAVA基础特性
- JAVA关键字
- javabean
- static
- 日期相关
- final
- interface
- 函数式接口
- JAVA异常
- 异常屏蔽
- try-with-resource资源泄露
- JAVA引用
- WeakReference
- SoftReference
- PhantomReference
- 位运算符
- try-with-resource语法糖
- JDK冷知识
- JAVA包装类
- JAVA基本类型与包装类
- java.lang.Boolean
- java.lang.Integer
- java.lang.Byte
- java.lang.Short
- java.lang.Long
- java.lang.Float
- java.lang.Double
- java.lang.Character
- 日期相关
- TemporalAdjusters
- String
- 字符串常量池
- String拼接
- String编译期优化
- StringBuilder&StringBuffer
- intern
- 注解
- java标准注解
- 内置注解
- 元注解
- 自定义注解
- 注解处理器
- JVM注解
- Java8 Annotation新特性
- 反射-Reflective
- Reflection
- Class
- Constructor
- Method
- javabean-property
- MethodHandles
- 泛型
- 类型擦除
- bridge-method
- Accessor&Mutator方法
- enum
- JAVA数组
- finalize方法
- JAR文件
- JAVA高级编程
- CORBA
- JMX
- SPI
- Java SPI使用约定
- ServiceLoader
- 实际应用
- IO
- 工具类
- JDK常用工具类
- Objects
- System
- Optional
- Throwable
- Collections
- Array
- Arrays
- System
- Unsafe
- Number
- ClassLoader
- Runtime
- Object
- Comparator
- VarHandle
- 数据结构
- 栈-Stack
- 队列(Queue)
- Deque
- PriorityQueue
- BlockingQueue
- SynchronousQueue
- ArrayBlockingQueue
- LinkedBlockingQueue
- PriorityBlockingQueue
- ConcurrentLinkedQueue
- 列表
- 迭代器
- KV键值对数据类型
- HashMap
- TreeMap
- Hash冲突
- ConcurrentHashMap
- JDK1.7 ConcurrentHashMap结构
- jdk7&jdk8区别
- 集合
- Vector
- Stack
- HashSet
- TreeSet
- ArrayList
- LinkedList
- ArrayList && LinkedList相互转换
- 线程安全的集合类
- 集合类遍历性能
- 并发容器
- CopyOnWriteArrayList
- ConcurrentHashMap
- 同步容器
- BitMap
- BloomFilter
- SkipList
- 设计模式
- 设计模式六大原则
- 单例模式
- 代理模式
- 静态代理
- 动态代理
- JDK动态代理
- cglib动态代理
- spring aop
- 策略模式
- SpringAOP策略模式的运用
- 生产者消费者模式
- 迭代器模式
- 函数式编程
- 方法引用
- 性能问题
- Lambda
- Lambda类型检查
- Stream
- findFirst和findAny
- reduce
- 原始类型流特化
- 无限流
- 收集器
- 并行流
- AOP
- 静态织入
- aspect
- aspect的定义
- AspectJ与SpringAOP
- 动态织入
- 静态代理
- 动态代理
- JDK动态代理
- CGLib动态代理
- Spring AOP
- SpringAOP五种通知类型
- @Before
- @AfterReturning
- @AfterThrowing
- @After
- @Around
- Aspect优先级
- SpringAOP切点表达式
- within
- execution
- 嵌套调用
- 系统优化与重构
- 重叠构造器模式
- 工具类构造器优化
- 常见面试题
- new Object()到底占用几个字节
- 访问修饰符
- cloneable接口实现原理
- 异常分类以及处理机制
- wait和sleep的区别
- 数组在内存中如何分配
- 类加载为什么要使用双亲委派模式,有没有什么场景是打破了这个模式
- 类的实例化顺序
- 附录
- JAVA术语
- FAQ
- 墨菲定律
- 康威定律
- 软件设计原则
- 阿姆达尔定律
- 字节码工具
- OSGI