EHCache简单使用 博客分类: java
转载:http://voyageivi.iteye.com/blog/1484715 感谢分享。
SQLCacheMySQLJVMXML .
关于缓存的话题,在坛子里已经有很多讨论,简单的来说,如果一个应用中80% 的时间内都在访问20% 的数据,那么,这时候就应该使用缓存了。
测试环境:
MySQL 5.0.22,
jdk1.6.0_07,
ehcache-1.6.0-beta2,
mysql-connector-java-3.1.14
测试表:
Sql代码
1.CREATE TABLE TEST
2.(
3.TEST_ID BIGINT,
4.TEST_NAME VARCHAR(50),
5.TEST_TIME TIMESTAMP,
6.TEST_VALUE DECIMAL(10, 3)
7.);
CREATE TABLE TEST
(
TEST_ID BIGINT,
TEST_NAME VARCHAR(50),
TEST_TIME TIMESTAMP,
TEST_VALUE DECIMAL(10, 3)
);
支持类:
Java代码
1.public class Util {
2.
3. public static Random rand = new Random();
4. public static String atoz = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
5.
6. public static String genString(int length) {
7. StringBuilder re = new StringBuilder(length);
8. re.append(atoz.charAt(rand.nextInt(52)));
9. for (int i = 0; i < length; i++) {
10. re.append(atoz.charAt(rand.nextInt(62)));
11. }
12. return re.toString();
13. }
14.
15. public static double genDouble() {
16. double d1 = 5120 * rand.nextDouble();
17. double d2 = 1024000 * rand.nextDouble();
18. return d1 + d2;
19. }
20.}
public class Util {
public static Random rand = new Random();
public static String atoz = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
public static String genString(int length) {
StringBuilder re = new StringBuilder(length);
re.append(atoz.charAt(rand.nextInt(52)));
for (int i = 0; i < length; i++) {
re.append(atoz.charAt(rand.nextInt(62)));
}
return re.toString();
}
public static double genDouble() {
double d1 = 5120 * rand.nextDouble();
double d2 = 1024000 * rand.nextDouble();
return d1 + d2;
}
}
插入测试数据:
Java代码
1.public static void traditionalInsert(int total) throws Exception {
2. Thread.sleep(3000);
3. Timestamp current = new Timestamp(System.currentTimeMillis());
4. String currentStr = dateFormat.format(current);
5. System.out.println(currentStr);
6. Connection conn = DriverManager.getConnection(dbURL, user, pass);
7. try {
8. long begin = System.currentTimeMillis();
9. conn.setAutoCommit(false);
10. String sql = "INSERT INTO TEST (TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE) VALUES (?, ?, ?, ?)";
11. PreparedStatement ps = conn.prepareStatement(sql);
12. for (int i = 1; i <= total; i++) {
13. ps.setLong(1, i);
14. ps.setString(2, Util.genString(33));
15. ps.setTimestamp(3, current);
16. ps.setBigDecimal(4, new BigDecimal(Util.genDouble()));
17. ps.addBatch();
18. if ((i % 500) == 0) {
19. ps.executeBatch();
20. }
21. }
22. ps.executeBatch();
23. conn.commit();
24. long end = System.currentTimeMillis();
25. System.out.printf("Count:%d Time:%d\n", total, (end - begin));
26. } catch (Exception ex) {
27. ex.printStackTrace();
28. conn.rollback();
29. } finally {
30. conn.close();
31. }
32. }
public static void traditionalInsert(int total) throws Exception {
Thread.sleep(3000);
Timestamp current = new Timestamp(System.currentTimeMillis());
String currentStr = dateFormat.format(current);
System.out.println(currentStr);
Connection conn = DriverManager.getConnection(dbURL, user, pass);
try {
long begin = System.currentTimeMillis();
conn.setAutoCommit(false);
String sql = "INSERT INTO TEST (TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE) VALUES (?, ?, ?, ?)";
PreparedStatement ps = conn.prepareStatement(sql);
for (int i = 1; i <= total; i++) {
ps.setLong(1, i);
ps.setString(2, Util.genString(33));
ps.setTimestamp(3, current);
ps.setBigDecimal(4, new BigDecimal(Util.genDouble()));
ps.addBatch();
if ((i % 500) == 0) {
ps.executeBatch();
}
}
ps.executeBatch();
conn.commit();
long end = System.currentTimeMillis();
System.out.printf("Count:%d Time:%d\n", total, (end - begin));
} catch (Exception ex) {
ex.printStackTrace();
conn.rollback();
} finally {
conn.close();
}
}
使用的javaBean:
Java代码
1.public class TEST implements Serializable {
2. private static final long serialVersionUID = 1L;
3. public Long TEST_ID;
4. public String TEST_NAME;
5. public Timestamp TEST_TIME;
6. public BigDecimal TEST_VALUE;
7. @Override
8. public String toString() {
9. return String.format("ID:%s,,,NAME:%s", TEST_ID, TEST_NAME);
10. }
11.}
public class TEST implements Serializable {
private static final long serialVersionUID = 1L;
public Long TEST_ID;
public String TEST_NAME;
public Timestamp TEST_TIME;
public BigDecimal TEST_VALUE;
@Override
public String toString() {
return String.format("ID:%s,,,NAME:%s", TEST_ID, TEST_NAME);
}
}
先试一下缓存到字典中:
Java代码
1.public static HashMap<Long, TEST> simpleCache() throws Exception {
2. HashMap<Long, TEST> cacheid = new HashMap<Long, TEST>();
3. Class.forName(dbDriver);
4. Connection conn = DriverManager.getConnection(dbURL, user, pass);
5. try {
6. long begin = System.currentTimeMillis();
7. Statement s = conn.createStatement();
8. String sql = "SELECT TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE FROM TEST";
9. ResultSet querySet = s.executeQuery(sql);
10. for (int i = 1; querySet.next(); i++) {
11. TEST curr = new TEST();
12. curr.TEST_ID = querySet.getLong(1);
13. curr.TEST_NAME = querySet.getString(2);
14. curr.TEST_TIME = querySet.getTimestamp(3);
15. curr.TEST_VALUE = querySet.getBigDecimal(4);
16. cacheid.put(curr.TEST_ID, curr);
17. }
18. long end = System.currentTimeMillis();
19. System.out.printf("Time:%d\n", (end - begin));
20. } catch (Exception ex) {
21. ex.printStackTrace();
22. } finally {
23. conn.close();
24. }
25. return cacheid;
26. }
public static HashMap<Long, TEST> simpleCache() throws Exception {
HashMap<Long, TEST> cacheid = new HashMap<Long, TEST>();
Class.forName(dbDriver);
Connection conn = DriverManager.getConnection(dbURL, user, pass);
try {
long begin = System.currentTimeMillis();
Statement s = conn.createStatement();
String sql = "SELECT TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE FROM TEST";
ResultSet querySet = s.executeQuery(sql);
for (int i = 1; querySet.next(); i++) {
TEST curr = new TEST();
curr.TEST_ID = querySet.getLong(1);
curr.TEST_NAME = querySet.getString(2);
curr.TEST_TIME = querySet.getTimestamp(3);
curr.TEST_VALUE = querySet.getBigDecimal(4);
cacheid.put(curr.TEST_ID, curr);
}
long end = System.currentTimeMillis();
System.out.printf("Time:%d\n", (end - begin));
} catch (Exception ex) {
ex.printStackTrace();
} finally {
conn.close();
}
return cacheid;
}
缓存到字典中,写法比较简单,使用方便,缺点就是缓存数据量比较少,一般缓存10W就有可能把jvm的缓存给占完了。用ehcache就可以解决缓存数据太少的问题。
一个简单的配置:
Xml代码
1.<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
2. xsi:noNamespaceSchemaLocation="ehcache.xsd">
3. <diskStore path="java.io.tmpdir"/>
4. <defaultCache
5. maxElementsInMemory="10000"
6. maxElementsOnDisk="0"
7. eternal="true"
8. overflowToDisk="true"
9. diskPersistent="false"
10. timeToIdleSeconds="0"
11. timeToLiveSeconds="0"
12. diskSpoolBufferSizeMB="50"
13. diskExpiryThreadIntervalSeconds="120"
14. memoryStoreEvictionPolicy="LFU"
15. />
16. <cache name="demoCache"
17. maxElementsInMemory="100"
18. maxElementsOnDisk="0"
19. eternal="false"
20. overflowToDisk="false"
21. diskPersistent="false"
22. timeToIdleSeconds="119"
23. timeToLiveSeconds="119"
24. diskSpoolBufferSizeMB="50"
25. diskExpiryThreadIntervalSeconds="120"
26. memoryStoreEvictionPolicy="FIFO"
27. />
28.</ehcache>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="ehcache.xsd">
<diskStore path="java.io.tmpdir"/>
<defaultCache
maxElementsInMemory="10000"
maxElementsOnDisk="0"
eternal="true"
overflowToDisk="true"
diskPersistent="false"
timeToIdleSeconds="0"
timeToLiveSeconds="0"
diskSpoolBufferSizeMB="50"
diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LFU"
/>
<cache name="demoCache"
maxElementsInMemory="100"
maxElementsOnDisk="0"
eternal="false"
overflowToDisk="false"
diskPersistent="false"
timeToIdleSeconds="119"
timeToLiveSeconds="119"
diskSpoolBufferSizeMB="50"
diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="FIFO"
/>
</ehcache>
Cache配置中的几个属性:
name:Cache的名称,必须是唯一的(ehcache会把这个cache放到HashMap里)。
maxElementsInMemory:内存中保持的对象数量。
maxElementsOnDisk:DiskStore中保持的对象数量,默认值为0,表示不限制。
eternal:是否是永恒数据,如果是,则它的超时设置会被忽略。
overflowToDisk:如果内存中数据数量超过maxElementsInMemory限制,是否要缓存到磁盘上。
timeToIdleSeconds:对象空闲时间,指对象在多长时间没有被访问就会失效。只对eternal为false的有效。默认值0,表示一直可以访问。
timeToLiveSeconds:对象存活时间,指对象从创建到失效所需要的时间。只对eternal为false的有效。默认值0,表示一直可以访问。
diskPersistent:是否在磁盘上持久化。指重启jvm后,数据是否有效。默认为false。
diskExpiryThreadIntervalSeconds:对象检测线程运行时间间隔。标识对象状态的线程多长时间运行一次。
diskSpoolBufferSizeMB:DiskStore使用的磁盘大小,默认值30MB。每个cache使用各自的DiskStore。
memoryStoreEvictionPolicy:如果内存中数据超过内存限制,向磁盘缓存时的策略。默认值LRU,可选FIFO、LFU。
获取配置中的demoCache:
Java代码
1.CacheManager manager = CacheManager.create("ehcache.xml");
2.Cache demo = manager.getCache("demoCache");
CacheManager manager = CacheManager.create("ehcache.xml");
Cache demo = manager.getCache("demoCache");
往cache中加入数据:
Java代码
1.public static void ehcache() throws Exception {
2. CacheManager manager = CacheManager.create("ehcache.xml");
3. manager.addCache("TEST_ID.TEST");
4. Cache cid = manager.getCache("TEST_ID.TEST");
5. Class.forName(dbDriver);
6. Connection conn = DriverManager.getConnection(dbURL, user, pass);
7. try {
8. long begin = System.currentTimeMillis();
9. Statement s = conn.createStatement();
10. String sql = "SELECT TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE FROM TEST";
11. ResultSet querySet = s.executeQuery(sql);
12. for (int i = 1; querySet.next(); i++) {
13. TEST curr = new TEST();
14. curr.TEST_ID = querySet.getLong(1);
15. curr.TEST_NAME = querySet.getString(2);
16. curr.TEST_TIME = querySet.getTimestamp(3);
17. curr.TEST_VALUE = querySet.getBigDecimal(4);
18. cid.put(new Element(curr.TEST_ID, curr));
19. }
20. long end = System.currentTimeMillis();
21. System.out.printf("Time:%d\n", (end - begin));
22. } catch (Exception ex) {
23. ex.printStackTrace();
24. } finally {
25. conn.close();
26. }
27. }
public static void ehcache() throws Exception {
CacheManager manager = CacheManager.create("ehcache.xml");
manager.addCache("TEST_ID.TEST");
Cache cid = manager.getCache("TEST_ID.TEST");
Class.forName(dbDriver);
Connection conn = DriverManager.getConnection(dbURL, user, pass);
try {
long begin = System.currentTimeMillis();
Statement s = conn.createStatement();
String sql = "SELECT TEST_ID,TEST_NAME,TEST_TIME,TEST_VALUE FROM TEST";
ResultSet querySet = s.executeQuery(sql);
for (int i = 1; querySet.next(); i++) {
TEST curr = new TEST();
curr.TEST_ID = querySet.getLong(1);
curr.TEST_NAME = querySet.getString(2);
curr.TEST_TIME = querySet.getTimestamp(3);
curr.TEST_VALUE = querySet.getBigDecimal(4);
cid.put(new Element(curr.TEST_ID, curr));
}
long end = System.currentTimeMillis();
System.out.printf("Time:%d\n", (end - begin));
} catch (Exception ex) {
ex.printStackTrace();
} finally {
conn.close();
}
}
这里在CacheManager中直接加入了一个叫TEST_ID.TEST的cache。因为只给了一个名字,所以系统会把defaultCache的设置给它clone一份。
使用方法,像字典一样使用就行:
Java代码
1.Cache cid = manager.getCache("TEST_ID.TEST");
2.Element e5120 = cid.get(new Long(5120));
3.System.out.println(e5120.getValue());
Cache cid = manager.getCache("TEST_ID.TEST");
Element e5120 = cid.get(new Long(5120));
System.out.println(e5120.getValue());
ehcache中数据是以java对象的形式存在的,使用了java的序列化保存到磁盘,所以保存的对象要实现Serializable接口。ehcache还可以支持分布式缓存。