专栏名称: ImportNew
伯乐在线旗下账号,专注Java技术分享,包括Java基础技术、进阶技能、架构设计和Java技术领域动态等。
目录
相关文章推荐
芋道源码  ·  孤陋寡闻了,原来 MySQL 还能这么写? ·  昨天  
芋道源码  ·  BigDecimal为什么可以不丢失精度? ·  2 天前  
芋道源码  ·  魔改 Arthas,又一款开源的线上问题定位神器 ·  3 天前  
芋道源码  ·  Nginx 实现动态封禁IP,详细教程来了 ·  3 天前  
51好读  ›  专栏  ›  ImportNew

ehcache 实现页面整体缓存和页面局部缓存

ImportNew  · 公众号  · Java  · 2017-01-21 22:59

正文

(点击上方公众号,可快速关注)


源:杨小剑,

www.cnblogs.com/jianjianyang/p/4953157.html

如有好文章投稿,请点击 → 这里了解详情


之前写过spring cache和ehcache的基本介绍和注解实现缓存管理,今天记录下web项目的页面缓存技术。


页面缓存是否有必要?


这样说吧,几乎所有的网站的首页都是访问率最高的,而首页上的数据来源又是非常广泛的,大多数来自不同的对象,而且有可能来自不同的db ,所以给首页做缓存是很必要的。那么主页的缓存策略应该怎样设计呢?我认为应该是某个固定时间之内不变的,比如说2分钟更新一次。那么这个缓存应该做在什么地方呢?让我们来看一下,当前我们的的应用的结构一般是是page-filter-action-service-dao-db ,这个过程中的- 的地方都是可以做缓存的地方,根据页面缓存的特征,应该把页面缓存做到尽量靠近客户的地方,就是在page 和filter 之间,这样的优点就是第一个用户请求之后,页面被缓存,第二个用户再来请求的时候,走到filter 这个请求就结束了,无需再走后面的action-service-dao-db 。带来的好处是服务器压力的减低和客户段页面响应速度的加快。了解了这些之后我们开始介绍重点。


ehcache页面缓存的特点:缓存中的元素是被压缩过的,如果客户浏览器支持压缩的话,filter 会直接返回压缩过的流,这样节省了带宽,把解压的工作交给了客户浏览器,如果客户的浏览器不支持gzip ,那么filter 会把缓存的元素拿出来解压后再返回给客户浏览器(大多数爬虫是不支持gzip 的,所以filter 也会解压后再返回流),这样做的优点是节省带宽,缺点就是增加了客户浏览器的负担


1 .SimplePageCachingFilter


它ehcache-web模块下页面缓存Filter的一个简单实现,适用于可以压缩的Http响应(response),如HTML、XML、JSON等。它会使用通过CacheManager的静态方法create创建的单例CacheManager,这样如果之前已经存在CacheManager的实例了的话,这里就会直接拿来用,而不会再创建了。所以这里一般默认情况下会取类根路径下的ehcache.xml文件来创建CacheManager,但如果我们的项目中整合了Ehcache和Spring,且在Spring配置文件中指定的Ehcache的配置文件不是默认位置的话,Spring将使用指定的配置文件优先初始化CacheManager,这样SimplePageCachingFilter中要使用CacheManager时就不会再初始化了,而是直接使用Spring初始化的。页面缓存使用的key是通过SimplePageCachingFilter的calculateKey()方法获取的。其内部逻辑是获取请求时的URI及后面的查询字符串作为key进行返回,如“/user/index.jsp?name=abc”,这使得它的应用范围非常广。它不依赖于主机名和端口号,这将使得它同样适用于有多个域或多个端口请求同样内容的情况。如果有需要,我们可以对calculateKey方法进行重写,从而实现我们自己的计算key的逻辑。这个是很有必要的,因为在项目中很多页面都使用AJAX,为保证JS请求的数据不被浏览器缓存,每次请求可能都会是不同的数参数。如果使用 SimplePageCachingFilter,那么每次生成的key都不一样,缓存就没有意义了。这种情况下,我们就会覆写 calculateKey()方法。


2 .SimpleCachingHeadersPageCachingFilter 提供HTTP缓存头信息,这个不介绍了用的很少。


3 .SimplePageFragmentCachingFilter


SimplePageCachingFilter适用于缓存整个页面的情况,如果只需要缓存某一个片段,如使用jsp:include包含的部分,这个时候就需要用SimplePageFragmentCachingFilter。


第一部分是页面整体缓存


第一步:首先配置ehcache.xml指定我们的SimplePageCachingFilter缓存  ,这里指定页面缓存的生命周期是60秒,还有timeToIdleSeconds的时间爱你是120秒,这里要注意下不要设置太长时间


   

           maxElementsInMemory="10"   

           maxElementsOnDisk="10"     

           eternal="false"

           overflowToDisk="false"

           timeToIdleSeconds="120"

           timeToLiveSeconds="60"

           memoryStoreEvictionPolicy="LFU">

   


第二步 : 在web.xml中添加页面缓存过滤器PageCachingFilter。


注意: 如果我们在ehcache.xml中命名的页面缓存名字为SimplePageCachingFilter时,我们再web.xml中的页面缓存过滤器的cacheName是可以不用定义的,因为它是默认的;如果不是SimplePageCachingFilter,这是我就必须指定cacheName了。


还有一点url-pattern的指定应该是/pageCacheController/testPageCache.do,而不是/testPageCache.do这个。


   

        PageCachingFilter

        net.sf.ehcache.constructs.web.filter.SimplePageCachingFilter

         

            cacheName  

            SimplePageCachingFilter  

         

      

   

        PageCachingFilter

        /pageCacheController/testPageCache.do

   


第三步 : 编写controller测试类


@Controller

@RequestMapping("pageCacheController")

public class PageCacheController {

 

    private final static Logger log = Logger.getLogger(PageCacheController.class);

 

    @RequestMapping("testPageCache")

    public ModelAndView testPageCache(){

        ModelMap model = new ModelMap();

        Date date = new Date();

        model.addAttribute("date", date.toLocaleString() );

        log.info("我来访问controller了");

        return new ModelAndView("testPageCache",model);

    }

}


String path = request.getContextPath();

String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";

%>

 

 

    测试