在项目中通常有生成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&amp;an&lt;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

FreeMarker在JAVA中应用入门的更多相关文章

  1. Java中Redis入门(1)

    Redis是一个开源的,先进的 key-value 存储可用于构建高性能,可扩展的 Web 应用程序的解决方案. Redis官方网网站是:http://www.redis.io/,如下: Redis ...

  2. java中多线程入门有趣介绍

    我们在网上可以看到所有有关于java的线程的基本概念的很多解释,不乏有很多详细经典的解释和代码解说.但是我们的很多初学者看完不能有一个直观的印象,特别是一些没有编程基础的学习者,很多时候要花很多时间去 ...

  3. [Freemarker] 在Java中简单实现对Freemarker的引用

    Demo目录结构 ├─src │ ├─main │ │ ├─java │ │ │ └─demo │ │ │ └─freemarker │ │ │ ├─main │ │ │ │ Test.java │ ...

  4. Java中Redis简单入门

    Redis是一个开源的,先进的 key-value 存储可用于构建高性能,可扩展的 Web 应用程序的解决方案. Redis官方网网站是:http://www.redis.io/,如下: Redis ...

  5. 【JAVA零基础入门系列】Day5 Java中的运算符

    运算符,顾名思义就是用于运算的符号,比如最简单的+-*/,这些运算符可以用来进行数学运算,举个最简单的栗子: 已知长方形的长为3cm,高为4cm,求长方形的面积. 好,我们先新建一个项目,命名为Rec ...

  6. 【JAVA零基础入门系列】Day9 Java中的那个大数值

    什么是大数值?用脚趾头想也知道,当然是"大"的数值(233).Java中有两个用于表示大数值的类,BigInteger和BigDecimal,那到底能表示多大的数值呢?理论上,可以 ...

  7. 【JAVA零基础入门系列】Day10 Java中的数组

    什么是数组?顾名思义,就是数据的组合,把一些相同类型的数放到一组里去. 那为什么要用数组呢?比如需要统计全班同学的成绩的时候,如果给班上50个同学的成绩信息都命名一个变量进行存储,显然不方便,而且在做 ...

  8. 【JAVA零基础入门系列】Day11 Java中的类和对象

    今天要说的是Java中两个非常重要的概念--类和对象. 什么是类,什么又是对象呢?类是对特定集合的概括描述,比如,人,这个类,外观特征上,有名字,有年龄,能说话,能吃饭等等,这是我们作为人类的相同特征 ...

  9. 第87节:Java中的Bootstrap基础与SQL入门

    第87节:Java中的Bootstrap基础与SQL入门 前言复习 什么是JQ? : write less do more 写更少的代码,做更多的事 找出所有兄弟: $("div" ...

随机推荐

  1. Oracle- 表的自增长创建

    Oracle创建自增长要先写序列还要去写触发器,不像MSSQLSERVER那样方便.但也是麻烦,记录如下: Oracle中,可以为每张表的主键创建一个单独的序列,然后从这个序列中获取自动增加的标识符, ...

  2. ECSHOP在线手册布局参考图--文章列表页 article_cat.dwt

        A.购物车 1,设置方法 程序自动读取购物车的商品数量 2,代码相关 cart.lbi 中 {insert_scripts files='transport.js'} <div clas ...

  3. Quartz管理类

    package com.sihuatech.project.task.manager; import java.text.ParseException; import org.quartz.CronT ...

  4. maven依赖传递关系

    一.maven 依赖传递规则 举个例子,比如A依赖B,B依赖C,那么A也是依赖C的.A是对B的直接依赖,A对C是传递依赖 ①.最短路劲原则 如,路劲一:A依赖B,B依赖C,C依赖D(1.0.0): 路 ...

  5. 在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke。

    本文转载:http://blog.csdn.net/playing9c/article/details/7471918 http://blog.csdn.net/beelinkerlidejun/ar ...

  6. iOS使用宏写单例

    本文只介绍ARC情况下的单例 过去一直背不下来单例如何写,就是知道这么回事,也知道通过宏来写单例,但是一直记不住,今天就来记录一下 - (void)viewDidLoad {     [super v ...

  7. iOS开发——Swift篇&单例的实现

    Swift实现单例模式 Swift实现单例模式 由于Swift语言弱化了struct和class之间的界限,这里我分别给出自己写的两种的单例实现 class版本: class SwiftSinglet ...

  8. 使用air进行移动app开发常见功能和问题(二)

    1.  Air如何判断android.ios 平台网络连接状态? Android,使用as3原生api: if(NetworkInfo.isSupported)//只有android支持 Networ ...

  9. 如何选择一个 Linux Tracer (2015)

    http://www.oschina.net/translate/choosing-a-linux-tracer

  10. SVN 之 去掉SVN管理标记

    当项目不需要SVN标志的时候, 方式一:很多人设置Windows显示隐藏文件,然后将项目中的所有.svn文件删除,这种方法是最笨的方法. 方式二:通过注册表方式 [1]新建一个文本文件.文本的内容为: ...