3

我们在项目中使用 ehcache 进行缓存。

import com.googlecode.ehcache.annotations.Cacheable;
// Other imports

@Component
public class Authenticator{
    @Cacheable(cacheName = "rest_client_authorized")
    public boolean isUserAuthorized(final String user, final String sessionId) {
        // Method code
    }
}

进入方法时没有缓存拦截器。到目前为止我们检查的内容:

  1. 我们不是从类内部调用这个方法,而是从外部调用。所以问题不在于导致绕过代理的内部调用。
  2. 我们为这个类添加了一个接口,并且我们改变了调用这个类的注入来使用接口表示而不是具体类。

我们以这种方式在应用程序上下文中定义了缓存管理器:

   <ehcache:annotation-driven cache-manager="ehCacheManager" />         
   <bean id="ehCacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
       <!-- use a share singleton CacheManager -->
       <property name="shared" value="true" />
   </bean>

缓存定义如下:

        <cache name="rest_client_authorized"
            eternal="false"
            maxElementsInMemory="50"
            overflowToDisk="false" diskPersistent="false"
            timeToIdleSeconds="0" timeToLiveSeconds="600"
            memoryStoreEvictionPolicy="LRU" />

当我们使用 Jconsole 测试缓存管理器时,我们可以看到缓存 *rest_auth_disabled* 存在并且为空。

任何关于为什么这不起作用的想法将不胜感激。谢谢!

更新(来自以下评论的汇总)

===========================================**

这是一个遗留代码,可以很好地与我提供的类和定义一起使用。我在这里谈论的方法是新的,但班上的其他人在过去确实有效。因此,我们正在努力了解发生了什么变化。我们还尝试将注释替换为 spring Cacheable,但仍然没有:/ 也许这取决于调用这个新方法的代码,它来自与我们用于其他方法的不同的 spring bean。但我仍然找不到问题。

还尝试在下面的答案之后返回布尔值而不是布尔值,但它没有用。我们有一个新的线索,可能与我们注入 bean 的方式有关(使用@Autowire)。如果确实如此,将更新。

4

4 回答 4

3

这个问题可能与 Springs 加载 bean 的顺序有关。尝试从 Authenticator 声明中删除 @Autowire 注释,并手动进行自动装配。就像是:

/**
 * Class that uses Authenticator
 */

 public class X {

    // Don't use @autowire here
    public Authenticator athenticator;

    public void doSomething() {
         manuallyAutowire();
    }

    public void manuallyAutowire() {
         if(authenticator == null) {
    authenticator = ApplicationContextUtils.getApplicationContext().
                            getBean(authenticator.class);
    }
}

在哪里

@Component
public class ApplicationContextUtils implements ApplicationContextAware {

    private static ApplicationContext ctx;

    @Override
    public void setApplicationContext(final ApplicationContext appContext) 
                                          throws BeansException {
         ctx = appContext;

    }

    public static ApplicationContext getApplicationContext() {
        return ctx;
    }
}
于 2013-04-18T08:58:35.977 回答
1

@Cacheable 中的参数cacheName的值应该和你的 application-context中<cache>声明的name属性的值相同

于 2013-04-10T12:53:35.493 回答
1

我认为你在这里混淆了东西 - 你已经使用过com.googlecode.ehcache.annotations.Cacheable,如果你想要 Springs 缓存支持,它实际上应该是org.springframework.cache.annotation.Cacheable。然后缓存拦截器应该干净地工作。

于 2013-04-10T13:19:44.680 回答
1

据我所知,Spring Ehcache 注释建议作为返回对象的两个参数都应该具有原始类型所缺少的equals()andhashCode()方法。

我不确定这个框架是否将原语转换为它们的包装变体(例如Integeror Boolean)。尝试返回包装变体Boolean而不是原始类型,看看它是否有效。

我不确定的另一件事是它如何(以及是否)处理final参数。如果我的第一个想法不起作用,请尽可能尝试删除final关键字,看看它是否有效。

于 2013-04-12T09:40:14.287 回答