Many Web applications are being rewritten from desktop applications; ideally, they should be as fast and scalable as the desktop versions. Almost all Web applications could benefit from a substantial increase in speed. Caching of frequently viewed data that changes infrequently is a powerful way to decrease wait time for users. A utility that easily handles the caching of data with a simple, nonintrusive API can help you accomplish this goal. The open source JCS, an Apache Jakarta project, is one such tool. This article explains how to configure and use JCS to cache data for your Web applications.
JCS is a caching system written in the Java language that you can use to create Java desktop and Web applications. It provides convenient mechanisms for storing data in the cache, getting data from the cache, removing data from the cache, and much more.
With JCS, you can store cached data in different data regions that you specify. JCS defines four types of core regions: memory, disk, lateral, and remote. You can use the core regions together to gain great flexibility in how and where you store your cache data. You can specify which region should be used first and when to fail over to another region.
The memory region is a pure-memory cache region that uses a Least Recently Used (LRU) algorithm. When the memory cache region becomes full, LRU removes the least recently used cached data first. This data region performs well, and most JCS users designate it as the default cache region to use first.
The disk region is a region for cache data on the Web server's file disk. To improve performance, JCS stores the cached data keys in memory while storing the actual cached data on the file disk. In a typical JCS configuration that uses the memory region first, any data that can't be held in the memory region is then written to the disk region.
The lateral cache region provides a configurable way to distribute the cached data across multiple servers. The cached data servers must have a port open to listen on, and a socket connection must be created. This region does face a potential problem because the region does not guarantee data consistency between caches. But if the region is used as designed, then this problem is unlikely to occur.
The remote region provides a cache region using a remote method invocation (RMI) API. The region uses a remote server that handles the cached data. The remote cached server can be used by multiple JCS client applications to store the cached data. Listeners are defined to collect the requests from the clients and the servers. This cache region helps to alleviate overhead with serialization and multiple connection points.
Configuring JCS is as simple as creating and populating a cache.ccf file. This file defines which regions your cache should use and the attributes or options of those regions. Tailoring the file to your application's needs is a convenient way to scale your cache quickly. I've made the following examples as simple as possible in order to display the main configuration points. You can specify a number of options and attributes to suit the configuration to exactly what you need.
Listing 1 displays the most basic cache.ccf file — a pure memory-cache configuration:
Listing 1. Basic configuration for JCS
|
You can see from the last line in Listing 1 that the configuration file specifies the memory cache as a LRUMemoryCache
. You can also see that the maximum number of objects kept in memory is set to 1000
.
Most applications would have a more comprehensive configuration for their caching system than the one in Listing 1. In the configuration in Listing 2, I use a memory region and a disk region while defining my own region (OUR_REGION
):
Listing 2. Regions defined in the configuration for JCS
jcs.default=DISK_REGION |
The first line in Listing 2 shows that the configuration is setting the default region to the DISK_REGION
. The DISK_REGION
is of type IndexedDiskCacheFactory
, and the file is specified on the disk as c:\jcs\disk_region. The second configuration group in Listing 2 defines my own region, for which I added some options. This type of configuration — one that uses both the memory region and a disk region while specifying a user-defined region — is a common one. The third configuration group in Listing 2 defines an auxiliary region.
JCS has two dependencies: concurrent
and commons-logging
. (Prior to JCS version 1.2.7.0, there were two additional dependencies: commons-collections
and commons-lang
.)
A good way to learn the basics of JCS is to view the API's most commonly used methods. The best place to start is with the initialization of the cache region itself. Initializing the JCS cache region object gives you access to almost all of the common methods you'll need. Listing 3 initializes the JCS object and gets an instance of the default cache region:
Listing 3. Retrieving the default cache region
// Initialize the JCS object and get an instance of the default cache region |
After retrieving the JCS instance, you can invoke the most needed methods. The put
method places a new object into the cache. All that is needed is a key
(first parameter) and a value
(second parameter). Listing 4 shows a basic example:
Listing 4. Setting a cached item
|
The example in Listing 4 uses string values for the parameters, but you can use any object.
Retrieving a cached object is as simple as using the get
method provided by JCS. Listing 5 shows a simple example. Again, the example uses a string parameter, but you can use any object.
Listing 5. Retrieving a cached item
// Retrieve a cached object |
A test for cache data validity is an additional method you might need when dealing with a caching system. In JCS, no test cache method is defined for simply testing for a cached item's existence. Instead, the get
method's return value can help. Listing 6 shows a way to achieve this necessary functionality:
Listing 6. Testing a cached item for validity
// Retrieve a cached object |
The last of the more common cache utilities you might need are for cleaning up JCS, the cached items, and the cache region after you are finished working with them. JCS provides a clear
method for removing all cached data from the cache region that it was called with and a remove
method for removing a specific cached item. A dispose
method is also available to dispose of the JCS region that was initialized. Listing 7 shows how to use these methods:
Listing 7. Cleaning up the cache region
// Dispose of a specific cached item |
One of the advantages of JCS over other caching systems (see Resources) is that it works well with objects. Most Web applications created with Java technology use an object-oriented approach. Caching of objects helps your application perform much better than it would if it continually retrieved objects piece by piece from a database, for example.
The first step in designing a simple object-oriented site around JCS starts with the object you need to store. For this example, I'll develop a basic blogging site, so I'll store the blog object itself. Listing 8 shows the BlogObject
class I'll use:
Listing 8.
BlogObject
classpackage com.ibm.developerWorks.objects; |
Once you have the object represented in a class, you next need a class to manage it. The manager handles all administrative aspects and caching functionality related to the blog objects. The manager, in this example, will handle three main tasks:
- Retrieving blog objects
- Setting blog objects in the cache
- Cleaning blog objects from the cache
The getBlog
method, shown in Listing 9, retrieves the blog object. The method first tries to attain the blog object from the cache. If the object is not in the cache, it will get it from another mechanism:
Listing 9. Retrieving a blog object via the blog manager
public BlogObject getBlog(int id) { |
In Listing 9, I use a database as the alternative mechanism for retrieving the object. When you retrieve the object from another mechanism, you should set that object to the cache so that the next retrieval can get it directly from the cache.
The setBlog
method, shown in Listing 10, places the blog object in the cache. This method is trivial because it simply creates a new blog object with the information passed in and then places it in the cache.
Listing 10. Placing a blog object in the cache via the blog manager
public boolean setBlog(int bId, String author, Date date, String title, String content) { |
The cleanBlog
method, shown in Listing 11, cleans either one specified blog or cleans all of the blogs out of the cache. The method uses JCS's remove
and clear
methods to clean up the cache objects.
Listing 11. Removing blog objects from the cache via the blog manager
public boolean cleanBlog(int blogId) { |
The previous classes demonstrate how straightforward caching objects with JCS is. With a simple representation of the object and a manager for that object, you have a powerful yet simple way to handle objects in your Web application.
JCS offers a great deal more than the basics I've shown you so far for adding caching to your applications. For example, it provides utilities for gathering metadata for cached objects and cache regions. You can easily retrieve:
- The cache key name
- The time the cached item was created
- The maximum time that the cache will live
- How much time until the cached item will expire
The example in Listing 12 shows how to retrieve metadata for cached items:
Listing 12. Retrieving metadata for cached items
try { |
Metadata for the cached items is useful, but it is also helpful to get metadata for each cache region. This information can let you know how much cached data is going into which region, including cache misses, cache hits, and cache updates. The example in Listing 13 shows how to gather this information:
Listing 13. Retrieving metadata for cache regions
try { |
Gathering metadata for cache regions and items helps you analyze which areas and items of your Web site need to be optimized. Metadata also helps you manage time-sensitive cached data. For example, you could use the maximum life and expiration time for each cached item to refresh cached data for specific users that need updated data.
JCS is a powerful and yet simple-to-use caching system for Java developers. It provides data caching for desktop and Web applications alike. The growth of desktop-like Web applications is requiring increased Web application speed and agility. Caching of data can aid in this effort. This overview has shown how to configure and use JCS. I have also covered the syntax needed for basic caching methods, caching of objects used in a common Web application, and the retrieval of cache metadata. You now have the initial knowledge of JCS to start developing your next Web site quickly with the power of data caching. You might also want to study several additional JCS areas that provide advanced functionality, such as HTTP Servlet access, JCS utlities, basic HTTP authentication, and additional auxiliary regions.
Learn
- Java Caching System: The official Web site of JCS.
- Java Caching System API: The API documentation for JCS.
- Ehcache: Ehcache is another general-purpose caching system for Java applications.
- OSCache: OSCache is a caching system that specifically deals with caching of JSP content.
- cache4j: A simple cache for Java objects.
- IBM Cache Advisor: Cache Advisor is an intelligent advising tool that generates optimal caching recommendations.
- Browse the technology bookstore for books on these and other technical topics.
- developerWorks Java technology zone: Find hundreds of articles about every aspect of Java programming.
No comments:
Post a Comment