使用Spring的AOP进行缓存在遭遇Hibernate应用时的注意点

4 篇文章 0 订阅
3 篇文章 0 订阅
Hibernate的实体有三种状态:Transient、Persistent、Detached,其中,只有处于Persistent的实体的变更能够被Hibernate监控到并进行相应的操作。

Hibernate对实体状态的操作,在使用了Spring的AOP进行缓存时就有了一些需要说明的问题了。 

ProductService是一个产品服务类,提供了获取所有产品getProducts()、获得指定编号产品getProduct(String)、更新指定产品价格updatePrice(String, int)三个接口。

ProductServiceAdvise是一个AOP的Advise,使用Around的方式进行拦截两个与ProductService同名的方法。简单的实例代码如下:
@Service @Transactional(propagation=Propagation.SUPPORTS)
public class ProductService {
 public Product getProduct(String productId) {
  return (Product) getSession().get(productId, Product.class);
 }
 
 @SuppressWarnings("unchecked")
 public List<Product> getProducts() {
  return getSession().getNamedQuery("GET_ALL_PRODUCTS").list();
 }
 
 @Transactional(propagation=Propagation.REQUIRED)
 public void updatePrice(String productId, int price) {
  Product product = this.getProduct(productId);
  product.setPrice(price);
 }
}

@Aspect @Service @Order(100)
public class ProductServiceAdvise {
 @Around("execution(* ProductService.getProduct(String))", argNames="productId")
 public Product getProduct(ProceedingJoinPoint pjp, String productId) throws Throwable {
  if (有缓存数据)
   return 缓存的product;
  
  Product product = (Product) pjp.proceed();
  // 增加到缓存容器中
  ....
  
  return product;
 }
}

正常情况下,ProductService.getProducts()和ProductService.getProduct(String)的调用都被相应的ProductServiceAdvise.getProducts()和ProductServiceAdvise.getProduct(String)给拦截掉,返回了ProductServiceAdvise缓存的数据,这种方式对于频繁的需要访问产品的应用提供了较好的优化,也是我们业务中常见的优化手段之一。

然而这种方式有一个陷进悄然潜伏着,不,应该说这种方式有个适用范围或者说潜规则或使用约定,即对被拦截返回的缓存数据应该进行只读操作,因为缓存返回的实体实例在Hibernate操作Session中将视为Detached,对这些实体的变更操作将不会被Session所监控到,自然也无法进行自动更新等操作了。以图解说明之:

AnotherService调用productService.getProduct(String)首先会被productServiceAdvise.getProduct(String)给拦截(图例1标注的地方),这里面有相应的判断决定是否需要调用productService.getProduct(String)(图例2标注的地方)。如果调用了,则返回的product是处于Persistent状态即可被当前Session管理的,如果没有被调用,则返回的product是detached的,是没法被当前Session管理的。

在ProductService的updatePrice(String, int)中,由于是使用了this.getProduct(String)去获取产品实例,使用this的调用在Spring的Proxy的AOP实现中,将不会被ProductServiceAdvise拦截,此时返回的始终是同一个Session中的Persistent实例,所以执行将完全正确。

但是,如果使用的AspectJ的编译方式的,由于在编译过程中在二进制代码中就加上了拦截,此时的调用将仍然被ProductServiceAdvise给拦截,执行的结果将处于一个不可预测状态了

  • 0
    点赞
  • 0
    收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
©️2022 CSDN 皮肤主题:大白 设计师:CSDN官方博客 返回首页
评论 1

打赏作者

MatrixII

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值