赛迪网 > IT技术 Java > 架构-与-模式
  IT资讯搜索
 
IT产品搜索
[程序开发][网管世界][网络安全][数据库技术]
[操作系统][嘉宾聊天·在线访谈][活动集锦]
[精彩专题][Symantec专区][订阅IT技术周刊]
[开发论坛][网管论坛][安全论坛][数据库论坛]
[操作系统论坛][Sybase专区][IBM dW技术专区]
[病毒求助][病毒与漏洞播报][文档·源码下载]

利用缓存机制快速读取XML文件中的数据

发布时间:2008.02.02 05:09     来源:赛迪网    作者:JooRoo

接到一个任务,让我做一个公司网站的后台管理系统。要求很简单,就一个新闻发布模块和一个招聘信息发布模块。但不能用DB,只能用文件存取的形式实现。

       不用考虑肯定是用XML文件进行数据的存取了,以前做毕设的时候也曾经实现过类似的功能,所以关于XML的读取并没有问题。关键是如果考虑到性能的问题就值得推敲一下了,我是新人,以前也没做过什么设计,所以做出的东西在一些人眼中可能会有些稚嫩,那也没关系,走自己的路让别人去说吧:)

       如果频繁解析文件,速度肯定受到影响,在文件非常大的情况下,甚至是无法忍受的。如果在服务器启动的时候先把文件中的数据封装成对象数组读入到缓存中,每次访问的时候先判断一下要访问的文件实体有没有被更新,如果没有被更新,就直接从缓存中将想要的数据读出来,当然如果文件被更新了,那只好老老实实的解析文件读出想要的数据。管理者对文件的修改次数毕竟是少数,更多的是访问者的访问次数。这样就能很大的提高了访问速度,代价是要占用一定的内存空间,但相比之下应该算小巫吧。

        下面把简单实现的几个Class说一说。

        一 首先实现一个Cache类,里面有读取对象的方法get(),如果文件没有被修改则直接从HashMap里面将对象取出,如果文件被修改则调用readObject()方法实现从文件中读出数据,并同时将读出的数据放入HashMap里,将原来的对象覆盖。这样下次再读数据的时候就可以从缓存中直接读到,并且保证是最新的数据。还有一个判断文件是否被修改的方法getModified();
代码实现如下:

import java.io.File;
import java.util.HashMap;

public class Cache {
    
    HashMap mapLastModified = new HashMap();
    HashMap mapValues = new HashMap();
    public Cache() {
        super();
    }    
    public Object get(String name, String path, Class clsParser, Class clsInstantiator, Class clsObj) {
        Object obj = null;
        String absPath = getClass().getResource(path).getPath();        
        Long modified = getModified(name, absPath);
        if (modified != null) {
            obj = readObject(absPath, clsParser, clsInstantiator, clsObj);
            
            mapLastModified.put(name, modified);
            mapValues.put(name, obj);
            System.out.println("get object from file");
        } else {
            obj = mapValues.get(name);
            System.out.println("get object from cache");
        }       
        return obj;
    }
    
    private Long getModified(String name, String path) {
        Long modified = new Long(new File(path).lastModified());
        Long saveModified = (Long) mapLastModified.get(name);
        if ((saveModified != null) && (saveModified.longValue() >= modified.longValue())) {
            modified = null;
        }        
        return modified;
    }
    
    private Object readObject(String path, Class clsParser, Class clsInstantiator, Class clsObj) {
        try {
            FileParser parser = (FileParser) clsParser.newInstance();
            Instantiator instantiator = (Instantiator) clsInstantiator.newInstance();

            Object config = parser.parse(path);            
            return instantiator.instantiate(clsObj, config);
            
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }       
        return null;
    }
}

二 解析XML文件的类XmlFileParser,
为了方便处理不同文件的解析,在这里先定义一个接口FileParser,XmlFileParser实现了它,如果还有诸如对其他种类文件的解析也可以实现它。
//FileParser.java
public interface FileParser {
    Object parse(String path);

}

//XmlFileParser.java
//采用Jdom的解析方式

import java.io.FileInputStream;
import java.io.IOException;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.input.SAXBuilder;

public class XmlFileParser implements FileParser {

 public XmlFileParser() {
  super();
 }

 public Object parse(String path) {
  
  FileInputStream fi = null;
  try {
   fi = new FileInputStream(path);
   SAXBuilder sb = new SAXBuilder();
   Document doc = sb.build(fi);
   Element root = doc.getRootElement();
   return root.getChildren();
  } catch (Exception e) {
   e.printStackTrace();
  } finally {
   try {
    fi.close();
   } catch (IOException e1) {
   }
  }  
 }
}

三 接下来是一个实例化处理的类ListTypeInstantiator,同样为了方便处理不同文件的实例化,在这里先定义一个接口Instantiator,ListTypeInstantiator实现了它。
//Instantiator.java
public interface Instantiator {
    Object instantiate(Class clazz, Object configuration);
}

//ListTypeInstantiator.java
import java.util.ArrayList;
import java.util.List;

import org.apache.commons.beanutils.BeanUtils;
import org.jdom.Element;

public class ListTypeInstantiator implements Instantiator {

 public ListTypeInstantiator() {
  super();
 }

 public Object instantiate(Class clazz, Object configuration) {
  List arr = new ArrayList();
  Object bean = null;
    
  List children = (List) configuration;
  Element child = null;

  List attributes = null;
  Element attribute = null;
  
  try { 
   for(int i=0; i    child = (Element) children.get(i);
    bean = clazz.newInstance();   
    attributes = child.getChildren();
    for(int j=0; j     attribute = (Element) attributes.get(j);
     BeanUtils.setProperty(bean, attribute.getName(), attribute.getText());
    }   
    arr.add(bean);
   }
  } catch(Exception e) {
   e.printStackTrace();
  }  
  return arr;
 }
}

四 另外还需要一个封装我想要数据形式的JavaBean,这里设为NewsBean{}.
//NewsBean.java
public class NewsBean {

 private Long id;
 private String newsTitle;
 private String newsContent;
 private String newsType;
 private String deployDate;
 private String cancelDate;
    
 public Long getId() {
  return id;
 }
 public void setId(Long id) {
  this.id = id;
 }   
 public String getNewsTitle() {
  return newsTitle;
 }
 public void setNewsTitle(String newsTitle) {
  this.newsTitle = newsTitle;
 }
 public String getNewsContent() {
  return newsContent;
 }
 public void setNewsContent(String newsContent) {
  this.newsContent = newsContent;
 }
 public String getNewsType() {
  return newsType;
 }
 public void setNewsType(String newsType) {
  this.newsType = newsType;
 }
 public String getDeployDate() {
  return deployDate;
 }
 public void setDeployDate(String deployDate) {
  this.deployDate = deployDate;
 }
 public String getCancelDate() {
  return cancelDate;
 }
 public void setCancelDate(String cancelDate) {
  this.cancelDate = cancelDate;
 } 
}

五 最后一步测试结果,将news.xml文件放到classes目录下。
//MainClass.java

import java.util.List;
public class MainClass{

 public static void main(String[] args) throws Exception {

  List news1 = null;
  List news2 = null;
  NewsBean bean = null;

  news1 = (List)Cache.get(
   "news", "/news.xml", 
      XmlFileParser.class, ListTypeInstantiator.class, NewsBean.class);  
  for (int i = 0; i < news1.size(); i++) {
   bean = (NewsBean) news1.get(i);
   System.out.println(bean.getId());
   System.out.println(bean.getNewsTitle());
   System.out.println(bean.getNewsContent());
   System.out.println(bean.getNewsType());
   System.out.println(bean.getDeployDate());
   System.out.println(bean.getCancelDate());
  } 
  news2 = (List)Cache.get(
   "news", "/news.xml", 
      XmlFileParser.class, ListTypeInstantiator.class, NewsBean.class);  
  for (int i = 0; i < news2.size(); i++) {
   bean = (NewsBean) news2.get(i);
   System.out.println(bean.getId());
   System.out.println(bean.getNewsTitle());
   System.out.println(bean.getNewsContent());
   System.out.println(bean.getNewsType());
   System.out.println(bean.getDeployDate());
   System.out.println(bean.getCancelDate());
  }
}
第一次会从文件中读出数据,第二次就会从缓存中读取了,试着多读几次速度明显快很多。
          (责任编辑:包春林)


[ 发表评论 ] 字体[  ] [ 打印 ] [ 进入博客 ] [ 进入论坛 ]  [ 推荐给朋友 ]
  相关文章
· Java程序类加载完全揭密 (02-01) · Java程序类加载完全揭密 (02-01)
· Java 在Client/Server 网络中的应用 (01-31) · J2ME综合--嵌入式LINUX中的JVM研究 (01-30)
· J2EE综合--关于权限设计的详细探讨 (01-30) · java学习-----理解面向对象的程序设计 (01-30)
· 开发框架:详细的spring入门给大家共享 (01-30) · 进阶:J2ME程序开发全方位基础讲解汇总 (01-30)
· 讨论有可能挑战Java开发优势的四种技术 (01-30) · 初学者入门--Structs中基本配置入门 (01-30)
  客户需求反馈表
* 姓  名:
更多资料  了解方案  认识厂商
* 单位名称:
* 联系电话:
* 电子邮件:
  赛迪推荐  
  手机·资费 ·新品·导购·评测·手机资费·宽带
手机搜索  诺基亚 N73 MOTO Z6
  IT产品 ·笔记本·台式机·服务器·打印·投影
IT产品搜索 
  IT技术 ·开发·网管·安全·数据库·操作系统
  信息化 ·热点·专题·访谈·周刊·方案案例
· 中小企业网站建设存在弊端 手机实名推波助澜
· CIO应如何克服三个关键错误信念 CIO委屈定理
· 五条黄金准则能够让CIO巧妙加薪 CIO焦虑调查
· 网上书店解决方案 深圳边检指挥中心ITSM项目
  IT博客 ·曾剑秋·项立刚·Java学习·网管
  IT技术论坛 ·开发·网管·安全·数据库·系统