`
fpjason
  • 浏览: 10879 次
  • 性别: Icon_minigender_1
  • 来自: 哈尔滨
最近访客 更多访客>>
社区版块
存档分类
最新评论

【转】使用缓存构建更快的 Web 应用程序

阅读更多
原文链接:《使用缓存构建更快的 Web 应用程序》

      使用 Java技术的 Web 开发人员可以使用缓存实用程序快速提升他们的应用程序的性能。Java 缓存系统(Java Caching System,JCS)是一个用于 Java 应用程序的强大分布式缓存系统,它是拥有简单 API 的高度可配置的工具。本文将概述 JCS 并展示如何使用它来提高 Web 应用程序的速度。

      许多 Web 应用程序会根据桌面应用程序重新编写;理想情况下,这些应用程序的速度和可伸缩性应该与桌面版本一样。几乎所有 Web 应用程序都可以从速度方面的增长获益。缓存被频繁查看但很少更改的数据是一种减少用户等待时间的有效方式。一个实用程序可以帮您实现这个目标,它使用简单易用的 API 来轻松处理数据缓存。开放源码 JCS(即一个 Apache Jakarta 项目)就是这样一种工具。本文将说明如何配置和使用 JCS 来缓存 Web 应用程序的数据。

      Java 缓存系统(Java Caching System,JCS)是一个用于 Java 应用程序的强大分布式缓存系统,它是拥有简单 API 的高度可配置的工具。本文将概述 JCS 并展示如何使用它来提高应用程序的速度。

      缓存被频繁查看但很少更改的数据是一种减少用户等待时间的有效方式。一个实用程序可以帮您实现这个目标,它使用简单易用的 API 来轻松处理数据缓存。开放源码 JCS(即一个 Apache Jakarta 项目)就是这样一种工具。本文将说明如何配置和使用 JCS 来缓存 Web 应用程序的数据。

JCS 概述

      JCS 是一个用 Java 语言编写的缓存系统,可以使用它来创建 Java 桌面和 Web 应用程序。它提供了在缓存器中存储数据、从缓存器中删除数据等方便机制。使用 JCS 可以在各种指定的数据区域中存储缓存数据。JCS 定义了 4 种类型的核心区域:内存区域、磁盘区域、外围区域和远程区域。可以结合使用这些核心区域以在如何存储缓存数据、将缓存数据存储在什么地方等方面获得更大的灵活性。您可以指定首次使用哪个区域,以及发生故障时转移到哪个区域。

内存区域

      内存区域是一个使用最近最少使用算法(Least Recently Used,LRU)的纯内存缓存区域。当内存缓存区域满时,LRU 会首先删除最近最少使用的缓存数据。这个数据区域执行良好,大多数 JCS 用户将它指定为最先使用的默认缓存区域。

磁盘区域

      磁盘区域是在 Web 服务器的文件磁盘上缓存数据。为了提高性能,JCS 在文件磁盘上存储实际缓存数据的同时,会在内存中存储缓存数据键。在首先使用内存区域的典型 JCS 配置中,任何无法在内存区域中保存的数据都会写入磁盘区域中。

外围区域

      外围区域提供一种可配置方式来在多台服务器之间分发缓存数据。缓存数据服务器必须有一个开放的用于侦听的端口,而且必须创建一个套接字连接。这个区域存在一个潜在问题,因为它不能保证各缓存之间的数据的一致性。但如果是按计划使用该区域,则不会出现这个问题。

远程区域

      远程区域提供了一个使用远程方法调用(RMI)API 的缓存区域。这个区域使用一台远程服务器处理缓存数据。这台远程缓存服务器可以被多个 JCS 客户端应用程序用于存储缓存数据。一些侦听器被定义用于收集来自客户端和服务器的请求。这个缓存区域帮助减少串行化和多个连接点的开销。

JCS 的可插入控制器

      JCS 通过组合缓存器(Composite Cache)让使用多个区域进行缓存存储变得很简单。组合缓存器为缓存区域提供了一个可插入控制器。组合缓存器仔细处理复杂的任务,即确定何时以及如何使用每个缓存区域。JCS 将完成大部分复杂的工作,开发人员只需关心获取和设置缓存。

JCS 配置

      配置 JCS 就是简单地创建和填充一个 cache.ccf 文件。这个文件定义缓存应该使用哪些区域,以及这些区域的属性或选项。根据应用程序的需求配置这个文件是一种快速扩展缓存的简便方式。您可以指定许多适合配置的选项和属性来满足需求。

清单 1 显示的是最基本的 cache.ccf 文件 — 一个纯内存缓存配置:


清单 1. JCS 的基本配置

jcs.default=jcs.default.cacheattributes=org.apache.jcs.engine.CompositeCacheAttributes
jcs.default.cacheattributes.MaxObjects=1000
jcs.default.cacheattributes.MemoryCacheName=
org.apache.jcs.engine.memory.lru.LRUMemoryCache

 
      从清单 1 中可以看出,该配置文件将内存缓存指定为一个 LRUMemoryCache。还可以看到,内存中能保存的对象的最大数量被设置为 1000。

      大多数应用程序的缓存系统配置要比清单 1 中复杂得多。在清单 2 的配置中,我在定义自己的区域(OUR_REGION)时使用了一个内存区域和一个磁盘区域:

清单 2. 在 JCS 配置中定义的区域

jcs.default=DISK_REGION
jcs.default.cacheattributes=org.apache.jcs.engine.CompositeCacheAttributes
jcs.default.cacheattributes.MaxObjects=1000
jcs.default.cacheattributes.MemoryCacheName=
  org.apache.jcs.engine.memory.lru.LRUMemoryCache

jcs.region.OUR_REGION=DISK_REGION
jcs.region.OUR_REGION.cacheattributes=org.apache.jcs.engine.CompositeCacheAttributes
jcs.region.OUR_REGION.cacheattributes.MaxObjects=1000
jcs.region.OUR_REGION.cacheattributes.MemoryCacheName=
  org.apache.jcs.engine.memory.lru.LRUMemoryCache
jcs.region.OUR_REGION.cacheattributes.UseMemoryShrinker=true
jcs.region.OUR_REGION.cacheattributes.MaxMemoryIdleTimeSeconds=3600
jcs.region.OUR_REGION.cacheattributes.ShrinkerIntervalSeconds=60
jcs.region.OUR_REGION.cacheattributes.MaxSpoolPerRun=500
jcs.region.OUR_REGION.elementattributes=org.apache.jcs.engine.ElementAttributes
jcs.region.OUR_REGION.elementattributes.IsEternal=false

jcs.auxiliary.DISK_REGION=org.apache.jcs.auxiliary.disk.indexed.IndexedDiskCacheFactory
jcs.auxiliary.DISK_REGION.attributes=
  org.apache.jcs.auxiliary.disk.indexed.IndexedDiskCacheAttributes
jcs.auxiliary.DISK_REGION.attributes.DiskPath=c:/jcs/disk_region
jcs.auxiliary.DISK_REGION.attributes.maxKeySize=100000

     清单 2 中的第一行表明该配置将默认区域设置为 DISK_REGION。DISK_REGION 是 IndexedDiskCacheFactory 类型,并且该文件在磁盘上指定为 c:"jcs"disk_region。清单 2 中的第二个配置组定义了我自己的区域,我为它添加了一些选项,这种类型的配置(在指定用户定义区域时同时使用内存区域和磁盘区域)是很常见的。清单 2 中的第 3 个配置组定义了一个辅助区域。

      JCS 有两个依赖项:concurrent 和 commons-logging(JCS 1.2.7.0 之前的版本中,还有两个其他依赖项:commons-collections 和 commons-lang)。

JCS 辅助插件

      除了 4 个核心缓存实现外,JCS 还提供了一些辅助插件,它们是区域可以使用的可选插件。这些辅助插件包括索引磁盘缓存(Indexed Disk Cache)、TCP 外围缓存(TCP Lateral Cache)和远程缓存服务器(Remote Cache Server)。例如,索引磁盘缓存允许在到达内存阈值时在磁盘上交换项目。这使得每个区域能更灵活地控制其缓存,提供一种类似于大多数操作系统所使用的虚拟内存的存储方法。cache.ccf 配置文件可以让每个辅助区域满足应用程序的需求。

JCS 的基本用法

      学习 JCS 基础知识的一个好方法是查看 API 最常用的方法。最好从初始化区域开始。初始化 JCS 缓存区域对象能使您访问大部分所需的常用方法。清单 3 初始化 JCS 对象并获得一个默认缓存区域实例:

清单 3. 检索默认缓存区域

// Initialize the JCS object and get an instance of the default cache region
JCS cache = JCS.getInstance("default");

     检索 JCS 实例后,可以调用最需要的方法。put 方法将一个新对象放入缓存中。接下来只需一个 key(第一个参数)和一个 value(第二个参数)。清单 4 显示一个基本示例:

清单 4. 设置缓存项

// Set up
String key = "key0";
String value = "value0";

// Place a new object in the cache
cache.put(key, value);

        清单 4 中的示例使用字符串值作为参数,但是您可以使用任何对象。检索缓存对象只不过是使用 JCS 提供的 get 方法。清单 5 显示了一个简单示例。同样,本例使用了一个字符串参数,但您可以使用任何对象。
清单 5. 检索缓存项

// Retrieve a cached object
String cachedData = (String)cache.get(key);

        测试缓存数据的有效性可能是处理缓存系统时需要使用的另一种方法。在 JCS 中,没有定义只测试缓存项是否存在的测试缓存方法。但是 get 方法的返回值可以帮助您。清单 6 显示了一种获得此必要功能的方式:
清单 6. 测试缓存项的有效性

// Retrieve a cached object
String cachedData = (String)cache.get(key);

// Check if the retrieval worked
if (cachedData != null) {
  // The cachedData is valid and can be used
  System.out.println("Valid cached Data: " + cachedData);
}

     最后需要几个用于在使用 JCS、缓存项和缓存区域后清除它们的常用缓存实用程序。JCS 提供了一种 clear 方法,用于从调用的缓存区域中删除所有缓存数据。此外,还提供了一个 remove 方法,用于删除指定缓存项。dispose 方法也可以处理初始化的 JCS 区域。清单 7 显示了如何使用这些方法:

// Dispose of a specific cached item
cache.remove(key);

// Dispose of all cache data
cache.clear();

// Dispose of the cache region
cache.dispose();

JCS 和 Java 对象

      JCS 优于其他缓存系统(请参阅 参考资料)的一个地方是它可以很好地使用对象。大多数 Web 应用程序是使用面向对象的方法通过 Java 技术创建的。例如,与连续从数据库中逐段检索对象相比,缓存对象使应用程序能够更好地执行。

设计一个简单的面向对象的 JCS 站点的第一个步骤是创建需要存储的对象。在本例中,我将开发一个基本 blogging 站点。清单 8 显示了我将使用的 BlogObject 类:

清单 8. BlogObject 类

import java.io.Serializable;
import java.util.Date;

public class BlogObject implements Serializable {
  private static final long serialVersionUID = 6392376146163510046L;
  private int blogId;
  private String author;
  private Date date;
  private String title;
  private String content;

  public BlogObject(int blogId, String author, Date date, String title, String content) {
    this.blogId = blogId;
    this.author = author;
    this.date = date;
    this.title = title;
    this.content = content;
  }

  public int getBlogId() {
    return this.blogId;
  }

  public String getAuthor() {
    return this.author;
  }

  public Date getDate() {
    return this.date;
  }

  public String getTitle() {
    return this.title;
  }

  public String getContent() {
    return this.content;
  }
}

        在一个类中表示对象后,接着还需要一个类来管理该对象。管理器处理所有与 blog 对象相关的管理和缓存功能。在本例中,管理器将处理三大任务:

1)检索 blog 对象

2)在缓存中设置 blog 对象

3)从缓存中清除 blog 对象

      如清单 9 所示,getBlog 方法检索 blog 对象。该方法首先试图从缓存获得 blog 对象。如果该对象不在缓存中,它将根据其他机制获取该对象:


清单 9. 通过 blog 管理器检索 blog 对象

public BlogObject getBlog(int id) {
  BlogObject blog = null;

  try {
    blogCache = JCS.getInstance(blogCacheRegion);
    blog = (BlogObject)blogCache.get(id);
  } catch (CacheException ce) {
    blog = null;
  }

  if (blog == null) {
    blog = DatabaseManager.getBlog(id);
    this.setBlog(
      blog.getBlogId(),
      blog.getAuthor(),
      blog.getDate(),
      blog.getTitle(),
      blog.getContent()
    );
  }

  return blog;
}

     在清单 9 中,我使用一个数据库作为检索 blog 对象的替代机制。根据另一种机制检索该对象时,应该将该对象设置为缓存,以便下一次检索可以直接从该缓存获取这个对象。

如清单 10 所示,setBlog 方法将 blog 对象放在缓存中。这个方法比较简单,因为它只是使用传入的信息创建一个新的 blog 对象,然后将这个对象放在缓存中。


清单 10. 通过 blog 管理器将 blog 对象放在缓存中

public boolean setBlog(int bId, String author, Date date, String title, String content) {
  BlogObject blog = new BlogObject(bId, author, date, title, content);

  try {
    blogCache = JCS.getInstance(blogCacheRegion);
    blogCache.put(bId, blog);
    return true;
  } catch (CacheException ce) {
    return false;
  }
}

     如清单 11 所示,cleanBlog 方法要么从缓存中清除一个指定的 blog,要么从缓存中清除掉所有 blog。这个方法使用 JCS 的 remove 和 clear 方法来清除缓存对象。


清单 11. 通过 blog 管理器从缓存中删除 blog 对象

public boolean cleanBlog(int blogId) {
  try {
    blogCache = JCS.getInstance(blogCacheRegion);
    blogCache.remove(blogId);
  } catch (CacheException ce) {
    return false;
  }
  return true;
}

public boolean cleanBlog() {
  try {
    blogCache = JCS.getInstance(blogCacheRegion);
    blogCache.clear();
  } catch (CacheException ce) {
    return false;
  }
  return true;
}

        前面的几个类展示了使用 JCS 缓存对象是很简单的。拥有对象管理器并使用简单的对象表示之后,您就获得一种在 Web 应用程序中处理对象的简单但强大的方法。

缓存元数据

      JCS 提供了更多方法,向应用程序添加缓存所用的方法只是其中的一小部分。例如,它提供了收集缓存对象和缓存区域元数据的实用程序。您可以轻松检索以下内容:

1)缓存键名称;2)创建缓存项的时间;3)缓存可以存在的最长时间;4)缓存过期时间.

清单 12 中的例子显示如何检索缓存项的元数据:


清单 12. 检索缓存项的元数据

try {
  JCSAdminBean admin = new JCSAdminBean();
  LinkedList linkedList = admin.buildElementInfo(regionName);
  ListIterator iterator = linkedList.listIterator();

  while (iterator.hasNext()) {
    CacheElementInfo info = (CacheElementInfo)iterator.next();
    System.out.println("Key: " + info.getKey());
    System.out.println("Creation Time: " + info.getCreateTime());
    System.out.println("Maximum Life (seconds): " + info.getMaxLifeSeconds());
    System.out.println("Expires in (seconds): " + info.getExpiresInSeconds());
  }
} catch (Exception e) {
}

     缓存项的元数据很有用,但获取各个缓存区域的元数据也很有帮助。这个信息让您知道缓存有多少数据,它们会进入哪个区域,包括缓存丢失、缓存提示和缓存更新。清单 13 中的示例显示如何获得此信息:


清单 13. 检索缓存区域的元数据

try {
  JCSAdminBean admin = new JCSAdminBean();
  LinkedList linkedList = admin.buildCacheInfo();
  ListIterator iterator = linkedList.listIterator();

  while (iterator.hasNext()) {
    CacheRegionInfo info = (CacheRegionInfo)iterator.next();
    CompositeCache compCache = info.getCache();
    System.out.println("Cache Name: " + compCache.getCacheName());
    System.out.println("Cache Type: " + compCache.getCacheType());
    System.out.println("Cache Misses (not found): " + compCache.getMissCountNotFound());
    System.out.println("Cache Misses (expired): " + compCache.getMissCountExpired());
    System.out.println("Cache Hits (memory): " + compCache.getHitCountRam());
    System.out.println("Cache Updates: " + compCache.getUpdateCount());
  }
} catch (Exception e) {
}

     收集缓存区域和项的元数据能帮助您分析 Web 站点的哪些区域和项目需要优化。元数据也能帮助您管理时间敏感型的缓存数据。例如,您可以使用每个缓存项的最长生命周期和过期时间来为需要更新数据的特定用户刷新缓存数据。

结束语

      JCS 是为 Java 开发人员提供的功能强大但简单易用的缓存系统。它为桌面和类似的 Web 应用程序提供数据缓存。类似桌面的 Web 应用程序的发展前景是提高速度和敏捷性。缓存数据对这些方面非常有益。本文概述如何配置和使用 JCS。此外,还讨论了基本缓存方法所需要语法,以及如何在常见 Web 应用程序中缓存对象和检索缓存元数据。解了 JCS 的基本知识之后,您现在可以利用数据缓存功能来开发下一个 Web 站点了。您还可以学习其他几个提供高级功能的 JCS 区域,比如 HTTP Servlet 访问、JCS 实用程序、基本 HTTP 验证和其他辅助区域。
分享到:
评论

相关推荐

    创建Web应用程序 使用.NET SDK创建Web应用程序非常必要

    使用.NET Framework SDK创建Web应用程序则需要手动完成更多步骤,这提供了更大的灵活性和控制权。首先,需要创建应用程序根目录,这是Web应用的基础。接着,根据需求创建子目录,用于组织内容和资源。然后,创建....

    基于FLEX技术构建可离线Web应用程序的研究与实现.pdf

    ### 基于FLEX技术构建可离线Web应用程序的研究与实现 #### 一、引言与背景 在计算机科学领域,特别是在Web应用程序的发展历程中,随着互联网技术的不断进步和用户需求的日益增长,传统的Web应用程序面临着诸多挑战...

    Node.js-Sapper一个构建高性能通用Web应用程序的框架

    **Node.js-Sapper:构建高性能通用Web应用程序的框架** Node.js是基于Chrome V8引擎的JavaScript运行环境,它以其异步非阻塞I/O模型在服务器端编程领域中独树一帜,使得开发者可以用JavaScript来编写后端代码。...

    使用Django和Python构建现代Web应用程序的5种最佳实践.docx

    利用ORM简化数据库操作,借助DRF构建RESTful API,使用WebSocket技术实现实时交互,以及通过缓存提升性能,这五个最佳实践将帮助开发者构建出高效、健壮且用户体验优秀的现代Web应用程序。遵循这些原则,开发者可以...

    现代 Web 应用程序的体系结构

    现代Web应用程序的体系结构是构建高效、可扩展和用户友好的网络应用的关键所在。随着互联网技术的不断发展,现代Web应用已经不再局限于简单的HTML和JavaScript页面,而是采用了多种技术和架构模式来提升性能、安全性...

    利用memcached构建高性能的Web应用程序[归纳].pdf

    【知识点详解】 1. **数据库存取瓶颈**...总结:利用Memcached构建高性能的Web应用程序,通过缓存层优化数据库存取,提高应用响应速度,降低数据库负载,同时要注意内存管理和缓存策略的设定,以实现系统的高效运行。

    react-一个基于17kbPreactRedux的渐进式Web应用程序

    标题 "react-一个基于17kbPreactRedux的渐进式Web应用程序" 暗示了这个项目是关于使用轻量级的Preact库和Redux状态管理器来构建一个渐进式Web应用程序(Progressive Web App,简称PWA)。Preact是一个与React高度...

    使用 HTML 5 支持移动 Web 应用程序离线工作

    本文将深入探讨如何使用 HTML5 的离线存储功能来构建支持移动 Web 应用程序离线工作的应用。 离线存储主要是通过两种技术实现的:`AppCache`(Application Cache)和`IndexedDB`。AppCache 提供了一个静态资源的...

    适用于 Web 应用程序的可扩展性能

    综上所述,通过使用 VMware vFabric GemFire 的 HTTP 会话管理和 Hibernate 缓存模块,开发人员不仅能够显著提升 Web 应用程序的性能,还能确保系统的高可用性和可扩展性,更好地满足不断增长的业务需求。

    .net 创立一个分布式web 应用程序

    首先,理解ASP.NET是微软提供的一个用于构建动态网站、Web应用程序和Web服务的开源框架。它基于.NET Framework,提供了一套强大的开发工具和服务器控件,简化了Web开发流程。 1. **架构设计**:在创建分布式Web应用...

    Node.js项目实践:构建可扩展的Web应用试读样章

    为了构建出具有实际使用价值的Node.js Web应用,开发过程中会用到各种组件、工具和模块库。如文中提到的Socket.IO、数据库连接模块、以及可能的前端组件和框架等。这些工具和模块库不仅仅是代码的复用,更重要的是...

    PhotoEditor一个基于React构建的单页图像编辑Web应用程序

    《基于React构建的单页图像编辑Web应用程序——深入解析PhotoEditor》 在现代Web开发领域,React作为一种流行的JavaScript库,已经成为了构建用户界面的首选工具。本文将详细探讨一款名为"PhotoEditor"的项目,这是...

    Sun Java Studio 的 Web 应用程序框架 概述.pdf

    **1.1 构建 Web 应用程序的挑战** 随着互联网的快速发展,构建高效、可扩展的 Web 应用程序成为了 IT 行业的一个重要议题。在 J2EE(Java 2 Platform, Enterprise Edition)出现之前,开发人员面临着一系列难题,如...

    Web应用程序中的数据缓存技术

    1. 数据缓存技术在Web应用程序中的重要性 随着网络技术的发展,Web支持服务的访问量呈现出稳步增长的趋势,如何处理巨大的数据量成为当前的一个重要议题。现有的数据缓存技术被分析,并针对其不足,提出了一个新的...

    详解大型网站web服务器缓存.zip

    应用程序缓存则更接近业务逻辑层面,通常是在应用程序内部实现,比如在数据库查询结果、计算结果等频繁使用的数据上进行缓存。这种缓存方式可以减少数据库访问,加快页面渲染速度。例如,Java的Spring框架中的...

    web项目页面缓存清除,不用每次刷新浏览器

    6. **Web App Manifest**:在创建Web应用程序时,manifest文件可以指示哪些资源应该被缓存,哪些资源需要在线使用。更新manifest文件也能触发Service Worker的更新,进而更新缓存。 7. **前端构建工具**:像Webpack...

    Web缓存策略:提升网站性能的高效途径

    在Web开发中,缓存是一种提高应用性能和用户体验的常用技术。通过合理利用Web缓存,可以减少服务器的负载,加快内容的加载速度,降低网络带宽的...掌握并应用这些策略,将有助于构建更快、更稳定、更可扩展的Web应用。

    缓存设计详解:低成本的高性能Web应用解决方案.doc

    首先,缓存的基本概念是临时存储数据,使得后续的请求能更快地获取信息。在互联网架构中,缓存主要存在于客户端浏览器、服务器端的转发代理和反向代理(如CDN)这三种位置。客户端浏览器缓存能够直接响应重复的请求...

    webapi接口缓存组件.rar

    缓存是提升应用程序性能的关键技术,它减少了对数据库或其他耗时资源的重复访问,降低了服务器负载。 在描述中提到的“接口添加缓存”,意味着这个组件旨在为Web API的各个接口提供缓存机制。缓存通常涉及将频繁...

Global site tag (gtag.js) - Google Analytics