在項目中通常有產生XML檔案發送到另一個系統的需求,簡單的辦法可以是用一個XML模板,通過Freemarker替換其中的'Mark'(${}),產生最終的XML檔案.
下面記錄了一下簡單的樣本步驟:
1,建立一個XML模板:
<?xml version="1.0" encoding="UTF-8"?><people xmlns:h="http://www.w3.org/TR/html4/"> <person id="000001" age="20"> <name> <family>${p.fname}</family> <given>${p.gname}</given> </name> <email>${p.email}</email> <link manager="${p.manager}" /> </person> </people>
2,在JAVA中將mark用值替換掉:下面是JAVA代碼
package com.test.xml.freemarker;public class ValueObject { private String fname; private String gname; private String email; private String manager; public String getFname() { return fname; } public void setFname(String fname) { this.fname = fname; } public String getGname() { return gname; } public void setGname(String gname) { this.gname = gname; } public String getEmail() { return email; } public void setEmail(String email) { this.email = email; } public String getManager() { return manager; } public void setManager(String manager) { this.manager = manager; } }public class Test { Configuration freeMarkerCfg = new Configuration(); Template template = null; public Test() { freeMarkerCfg.setClassForTemplateLoading(getClass(),""); freeMarkerCfg.setObjectWrapper(new DefaultObjectWrapper()); try { template = freeMarkerCfg.getTemplate("test.xml"); } catch (IOException e) { // TODO e.printStackTrace(); } } public void generateRequest(ValueObject obj) { String reqFileName = "D:\\temp\\test.xml"; try { Map<String, Object> parameters = new HashMap<String, Object>(); parameters.put("p", obj); OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName), "UTF-8"); template.process(parameters, writer); writer.flush(); } catch (IOException e) { e.printStackTrace();// TODO } catch (TemplateException e) { e.printStackTrace();// TODO } } public static void main(String[] args) { ValueObject val = new ValueObject(); val.setFname("Yao"); val.setGname("Yorker"); val.setEmail("test@mail.com"); val.setManager("manager"); new Test().generateRequest(val); }}
XML模板和class在package "com.test.xml.freemarker".
***在模板中通過${val}指定的值,如果在處理的時候val值為null,會出現下面的異常:
freemarker.core.InvalidReferenceException: Expression valis undefined on line 46, column 63 in test.ftl
FreeMarker這麼做的本意是防止在傳值的地方(Data Model)寫入的參數,如想傳入 parameters.put("abc", obj);,但是寫成了parameters.put("abcd", obj);
但是有的時候,裡面有的值不是必須有值的,可以通過${val!""}來繞過這個異常.${val!""}的意思是如果val為null,取值"".
***Freemarker對XML檔案中特殊字元的處理:通過<#escape>
<#escape x as x?xml> <person id="000001" age="20"> <name> <family>${p.fname}</family> <given>${p.gname}</given> </name> <email>${p.email}</email> <link manager="${p.manager}" /> </person> </#escape>
val.setManager("m&an<ager"); 傳入的值就按照XML規範儲存到XML檔案 link manager="m&an<ager" />
***迴圈處理 <#list>
<#list people as p> <person id="000001" age="20"> <name> <family>${p.fname}</family> <given>${p.gname}</given> </name> <email>${p.email}</email> <link manager="${p.manager}" /> </person> </#list> List<ValueObject> pList = new ArrayList<ValueObject>(); ValueObject val = null; val = new ValueObject(); val.setFname("Yao"); val.setGname("Yorker"); val.setEmail("test@mail.com"); val.setManager("m&an<ager"); pList.add(val); val = new ValueObject(); val.setFname("J"); val.setGname("Jeremy"); val.setEmail("test1@mail.com"); val.setManager("m&an<ager"); pList.add(val); parameters.put("people", pList);
***分支處理<#if>,根據值對模板做不同的輸出.
<#if p.level == "L1">
<l1tag>xxx</l1tag>
</#if>
***對產生超大檔案的測試:
public void generateRequest(List<ValueObject> pList) { String reqFileName = "D:\\temp\\test.xml"; try { Map<String, Object> parameters = new HashMap<String, Object>(); parameters.put("people", pList); OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName), "UTF-8"); template.process(parameters, writer); writer.flush(); } catch (IOException e) { e.printStackTrace();// TODO } catch (TemplateException e) { e.printStackTrace();// TODO } } public static void testLargeVolumn(){ List<ValueObject> pList = new ArrayList<ValueObject>(); ValueObject val = null; for(int i=0;i<400000;i++){ val = new ValueObject(); val.setFname("Yao"+i); val.setGname("Yorker"); val.setEmail("test@mail.com"); val.setManager("m&an<ager"); val.setLevel("L"+i); pList.add(val); } new Test().generateRequest(pList); }
一次性輸出,本例在達到迴圈400000次的時候異常:java.lang.OutOfMemoryError: Java heap space
解決辦法:分多次輸出到最終的輸出檔案中.
1,將模板需要迴圈的部分單獨做成一個模板.
先一次輸出迴圈模板前面的部分,分多次輸出迴圈部分,一次輸出迴圈部分之後的部分.
樣本如下,(這裡由於迴圈之前部分和之後部分不涉及模板替換,直接用JAVA代碼輸出)
模板需要迴圈輸出的部分:
<#escape x as x?xml> <#list people as p> <person id="000001" age="20"> <name> <family>${p.fname}</family> <given>${p.gname}</given> </name> <email>${p.email}</email> <link manager="${p.manager}" /> <#if p.level == "L1"> <l1tag>xxx</l1tag> </#if> </person> </#list> </#escape>
JAVA代碼:
public class LargeVolumnTest { Configuration freeMarkerCfg = new Configuration(); Template template = null; public LargeVolumnTest() { freeMarkerCfg.setClassForTemplateLoading(getClass(),""); freeMarkerCfg.setObjectWrapper(new DefaultObjectWrapper()); try { template = freeMarkerCfg.getTemplate("testL.xml"); } catch (IOException e) { // TODO e.printStackTrace(); } } public void generateRequestHeader() { String reqFileName = "D:\\temp\\test.xml"; try { OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName), "UTF-8"); writer.write("<?xml version=\"1.0\" encoding=\"UTF-8\"?>"); writer.write("<people xmlns:h=\"http://www.w3.org/TR/html4/\">"); writer.flush(); } catch (IOException e) { e.printStackTrace();// TODO } } public void generateRequestTail() { String reqFileName = "D:\\temp\\test.xml"; try { OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName,true), "UTF-8"); writer.write("</people>"); writer.flush(); } catch (IOException e) { e.printStackTrace();// TODO } } public void generateRequest(List<ValueObject> pList) { String reqFileName = "D:\\temp\\test.xml"; try { Map<String, Object> parameters = new HashMap<String, Object>(); parameters.put("people", pList); OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName,true), "UTF-8"); template.process(parameters, writer); writer.flush(); } catch (IOException e) { e.printStackTrace();// TODO } catch (TemplateException e) { e.printStackTrace();// TODO } } public void testLargeVolumn(){ generateRequestHeader(); List<ValueObject> pList = new ArrayList<ValueObject>(); ValueObject val = null; int i=0; for(;i<500000;i++){ val = new ValueObject(); val.setFname("Yao"+i); val.setGname("Yorker"); val.setEmail("test@mail.com"); val.setManager("m&an<ager"); val.setLevel("L"+i); pList.add(val); if(i%10000==0){ generateRequest(pList); pList.clear(); } } if(i%10000!=0){ generateRequest(pList); } generateRequestTail(); } public static void main(String[] args) { new LargeVolumnTest().testLargeVolumn(); }}
注意在輸出迴圈部分和結尾部分是通過追加到檔案的方式:OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(reqFileName,true), "UTF-8");
最終產生的XML文檔有90M,沒有報異常.
這裡有FreeMarker的中文manual :http://download.csdn.net/user/kkdelta