歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

在 Spring 3.1 中使用 @Cacheable 實現緩存

在軟件開發中使用緩存已經有一個非常久的歷史了。緩存是一種很好的設計思想,一旦你用了他,你將會發現他確實很有用。Spring3.1版本的核心對緩存做了實現。在Java推出Annotation特性之前,實現緩存的一個難點在於它與業務邏輯代碼的耦合性太強。

然而,Spring3.1中使用@Cacheable 和@CacheEvict實現緩存在某種程度上解決了這個問題,基本思想是在方法加上@Cacheable注解,這個方法的返回值將具有緩存特性。

@Cacheable注解可以用在方法或者類級別。當他應用於方法級別的時候,就是如上所說的緩存返回值了。當應用在類級別的時候,這個類的所有方法的返回值都將被緩存。

@Cacheable(value = "employee")
public class EmployeeDAO {

  public Person findEmployee(String firstName, String surname, int age) {

    return new Person(firstName, surname, age);
  }

  public Person findAnotherEmployee(String firstName, String surname, int age) {

    return new Person(firstName, surname, age);
  }
}

@Cacheable注解有三個參數,value是必須的,還有key和condition。第一個參數,也就是value指明了緩存將被存到什麼地方。

@Cacheable(value = "employee")
 public Person findEmployee(String firstName, String surname, int age) {

   return new Person(firstName, surname, age);
 }

上面的代碼保證findEmployee的返回值Person對象將被存儲在"employee"中。

任何存儲在緩存中的數據為了高速訪問都需要一個key。Spring默認使用被@Cacheable注解的方法的簽名來作為key,當然你可以重寫key,自定義key可以使用SpEL表達式。

@Cacheable(value = "employee", key = "#surname")
   public Person findEmployeeBySurname(String firstName, String surname, int age) {

    return new Person(firstName, surname, age);
  }

在findEmployeeBySurname()的注解中"#surname"是一個SpEL表達式,他將使用findEmployeeBySurname()方法中的surname參數作為key。

@Cacheable的最後一個參數是condition(可選),同樣的,也是引用一個SpEL表達式。但是這個參數將指明方法的返回結果是否被緩存。

@Cacheable(value = "employee", condition = "#age < 25")
 public Person findEmployeeByAge(String firstName, String surname, int age) {

   return new Person(firstName, surname, age);
 }

上面的例子中,只有年齡小於25的時候才被緩存。

在快速看完了如何使用緩存後,我們接下來看看緩存帶來的效果。

<A href="http://my.oschina.net/test45" target=_blank rel=nofollow>@Test</A> 
  public void testCache() {

    Person employee1 = instance.findEmployee("John", "Smith", 33);
    Person employee2 = instance.findEmployee("John", "Smith", 33);

    assertEquals(employee1, employee2);
  }

上面的例子很簡單,第一次調用findEmployee,findEmployee方法將被執行,Spring將他的返回值一個person對象存入緩存。第二次調用findEmployee的時候findEmployee將不被執行,Spring直接將緩存中的數據作為返回值返回。所以employee1 和employee2引用了同樣的對象。


而下面的例子中,我們將年齡小於25作為緩存條件,就將得到不同的結果。

<A href="http://my.oschina.net/test45" target=_blank rel=nofollow>@Test</A>
 public void testCacheWithAgeAsCondition() {

   Person employee1 = instance.findEmployeeByAge("John", "Smith", 33);
   Person employee2 = instance.findEmployeeByAge("John", "Smith", 33);

   assertEquals(employee1, employee2);
 }

下面的例子我們在findEmployeeBySurname的方法的注解中自定義了key,我們使用了自定義的key生成方式,以確保不同的surname將會指向不同的人。看下面的程序

<A href="http://my.oschina.net/test45" target=_blank rel=nofollow>@Test</A> 
 public void testCacheOnSurnameAsKey() {

   Person employee1 = instance.findEmployeeBySurname("John", "Smith", 22);
   Person employee2 = instance.findEmployeeBySurname("Jack", "Smith", 55);

   assertEquals(employee1, employee2);
 }
我們想找到兩個不同的人,但是兩個人的surname是相同的,你將發現兩次調用返回了相同的結果,這不是Spring的問題,而是我們的cache key的生成方式有問題。所以在我們定義key的時候要小心注意key的生成策略,避免造成這種問題。
最後總結一下流程,當執行到一個被@Cacheable注解的方法時,Spring首先檢查condition條件是否滿足,如果不滿足,執行方法,返回;如果滿足,在value所命名的緩存空間中查找使用key存儲的對象,如果找到,將找到的結果返回,如果沒有找到執行方法,將方法的返回值以key-對象的方式存入value緩存中,然後方法返回。
上文僅僅是@Cacheable的使用方法,但是我們怎麼使用@CacheEvict注解來清除緩存呢?另外,還有一個問題,如何選擇一個緩存的實現,並配置Spring的緩存呢?欲知後事如何,且聽下回分解。
OSChina 原創翻譯/ 原文鏈接
Copyright © Linux教程網 All Rights Reserved