利用緩衝機制快速讀取XML檔案中的資料

來源:互聯網
上載者:User
xml|緩衝|資料 接到一個任務,讓我做一個公司網站的後台管理系統。要求很簡單,就一個新聞發布模組和一個招聘資訊發布模組。但不能用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<children.size(); i++) {
    child = (Element) children.get(i);
    bean = clazz.newInstance();   
    attributes = child.getChildren();
    for(int j=0; j<attributes.size(); 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());
  }
}
第一次會從檔案中讀出資料,第二次就會從緩衝中讀取了,試著多讀幾次速度明顯快很多。
 


相關文章

Alibaba Cloud 10 Year Anniversary

With You, We are Shaping a Digital World, 2009-2019

Learn more >

Apsara Conference 2019

The Rise of Data Intelligence, September 25th - 27th, Hangzhou, China

Learn more >

Alibaba Cloud Free Trial

Learn and experience the power of Alibaba Cloud with a free trial worth $300-1200 USD

Learn more >

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。