另类:JAVA用freemarker生成复杂Excel。(freemarker)

上次写了一篇关于生成Word文档:http://blog.csdn.net/zhanwentao2/article/details/7255432


在生成Excel的时候,大多时候都是使用poi,jxl等进行的,但是对于复杂的Excel来说,这个工作量是非常的大的,而且,对于我这么懒的人来说,这是相当痛苦的一件事情,所以,我不得不找找有没有简单一点的办法来搞定这个事情。我想到了上次用freemarker成功生成了Word文档。我就想可不可以也生成Excel呢?然后我就尝试。结果真行,不过,和生成Word文档有一些差别,好了,不废话了,上图。

上一张比较简单的Excel图。


这个,就是Excel的一个结构了,接下了,将它另存为“XML 电子表格2003(*.xml)”

至如如何将数据准确的填充到相应的地方,看上一篇生成Word文档的小技巧,就是在需要填充数据的地方加上一个标记,或者写拼音。

下图是生成的Xml



我想,这个结构大家应该能一目了然了,row:代表一行,cell代表一个单元格.

用过poi,jxl的朋友应该非常的熟悉了.

有的朋友可能打开的时候,xml非常的凌乱,找不到需要修改的地方,上面所说的小技巧就起到了作用,按照拼音或者你独特的标识找到需要修改的位置.

在需要填充数据的地方写上freemarker模块语句。例如:将类型改为"${type}" ,双引号别也拷贝过去了,呵呵.

还有,这一行可能有很多行,所有需要一个迭代的标签,在上一篇我也说到过的.

<#list sequence as item>

</#list>

上面的语法格式中,sequence就是一个集合对象,也可以是一个表达式,但该表达式将返回一个集合对象,

而item是一个任意的名字,就是被迭代输出的集合元素.此外,迭代集合对象时,还包含两个特殊的循环变量:
item_index:当前变量的索引值
item_has_next:是否存在下一个对象
也可以使用<#break>指令跳出迭代

其实这些在http://www.havenliu.com/goodarticle/420.html里面都是有的,有时间的朋友可以看看.有许多有意思的东西.


回到正题,将你需要循环遍历的一行剪切到<#list></#list>中间.

好,做完这一步,就可以将它的后缀名改为:ftl了。

工作做到这一步已经差不多了,代码的话,可以看上一篇.算了,负点责任,将代码贴上并写一些注释吧.

package com;  
  
import java.io.BufferedWriter;  
import java.io.File;  
import java.io.FileNotFoundException;  
import java.io.FileOutputStream;  
import java.io.IOException;  
import java.io.OutputStreamWriter;  
import java.io.Writer;  
import java.util.ArrayList;  
import java.util.HashMap;  
import java.util.List;  
import java.util.Map;  
  
import freemarker.template.Configuration;  
import freemarker.template.Template;  
import freemarker.template.TemplateException;  
  
public class WordTest {  
      
    private Configuration configuration = null;  
      
    public WordTest(){  
        configuration = new Configuration();  
        configuration.setDefaultEncoding("UTF-8");  
    }  
      
    public static void main(String[] args) {  
        WordTest test = new WordTest();  
        test.createWord();  
    }  
      
    public void createWord(){  
        Map<String,Object> dataMap=new HashMap<String,Object>();  
        getData(dataMap);  
        configuration.setClassForTemplateLoading(this.getClass(), "/com");  //FTL文件所存在的位置  
        Template t=null;  
        try {  
            t = configuration.getTemplate("wordModel.ftl"); //文件名  
        } catch (IOException e) {  
            e.printStackTrace();  
        }  
        File outFile = new File("O:/outFilessa"+Math.random()*10000+".xls");  //生成文件的路径
        Writer out = null;  
        try {  
            out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile)));  
        } catch (FileNotFoundException e1) {  
            e1.printStackTrace();  
        }  
           
        try {  
            t.process(dataMap, out);  
        } catch (TemplateException e) {  
            e.printStackTrace();  
        } catch (IOException e) {  
            e.printStackTrace();  
        }  
    }  
    //这里赋值的时候需要注意,xml中需要的数据你必须提供给它,不然会报找不到某元素错的.
    private void getData(Map<String, Object> dataMap) {  
        dataMap.put("title", "标题");  
        dataMap.put("year", "2012");  
        dataMap.put("month", "2");  
        dataMap.put("day", "13");  
        dataMap.put("auditor", "鑫");  
        dataMap.put("phone", "xxxxxxxxxxxxx");  
        dataMap.put("weave", "文涛");  
//      dataMap.put("number", 1);  
//      dataMap.put("content", "内容"+2);  
          
        List<Map<String,Object>> list = new ArrayList<Map<String,Object>>();  
        for (int i = 0; i < 10; i++) {  
            Map<String,Object> map = new HashMap<String,Object>();  
            map.put("number", i);  
            map.put("content", "内容"+i);  
            list.add(map);  
        }  
          
          
        dataMap.put("list", list);  
    }  
}  
好了,大致就这样了.

如何程序没报错的话,那么你的Excel已经成功生成出来了,但是.

有时候会打不开?表出错?


这是因为Excel在生成xml的时候,可能是为了不浪费资源吧,在生成时候,会指定Excel有多少行,如果超出了这个长度的话,它就会编译不通过.报错.

问题很好解决.

针对这2个问题,需要修改2个不同的地方.

1:工作表的到设置.就是Sheet了.ok,打开xml,搜索"Worksheet",如果你有多个工作表的话,找到你自己要的.

修改它的属性值"ss:ExpandedRowCount",看到这,应该懂了吧?把它设置大一点,或者通过程序传个值给它也可以.这个问题就解决了.

2:表,如果你有耐性,可以根据它给你的log去看看,一看也就知道了.不过不是很好找,呵呵.

搜索最后一个Row,修改它的属性值"ss:Index".


好了,基本上没问题了,就这样吧.如果有什么不懂的,请留言.





  • 15
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 67
    评论
SpringBoot_Freemarker生成Word_多个表格+两层嵌套循环; 步骤说明: 1.用Microsoft Office Word打开word原件;将文档中需要动态生成的内容,替换为属性名 ${name} 2.另存为,选择保存类型Word 2003 XML 文档(*.xml) 3.用Firstobject free XML editor打开文件,选择Tools下的Indent【或者按快捷键F8】格式化文件内容。左边是文档结构,右边是文档内容; 4. 文档生成后有时需要手动修改,查找第一步中设置的属性名,可能会产生类似${n.....ame}类似的样子,我们将将名字中间的标签删掉,恢复为${name} 5. word模板中有表格,需要循环的位置, 用 标签将第二对 标签(即除表头的w:tr标签后的一对)包围起来 同时表格内的属性例如${name},在这里需要修改为${user.name} (userList是集合在dataMap中的key, user是集合中的每个元素, 类似), 如图: PLUS:若表格之外还有嵌套的循环,也需要用,注意这里的标签不要和某对其他标签交叉,不可以出现这种 6. 标识替换完之后,另存为.ftl后缀文件即可。 代码里是相对有一丢丢复杂的,两层嵌套循环; 总(dataMap) deptName 部门名 list(Table)表的集合 table1(map) table-名字 ${map.table} tableName-中文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段中文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table2 table-名字 ${map.table} tableName-中文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段中文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table3 ……
Java 中使用 Freemarker 生成 XML 文件,需要先添加依赖。 Maven 依赖: ```xml <dependency> <groupId>org.freemarker</groupId> <artifactId>freemarker</artifactId> <version>2.3.31</version> </dependency> ``` Gradle 依赖: ```groovy implementation 'org.freemarker:freemarker:2.3.31' ``` 接下来,我们可以编写 Freemarker 模板文件 `template.ftl`,用于生成 XML。 示例模板文件: ```xml <?xml version="1.0" encoding="UTF-8"?> <root> <user id="${user.id}" name="${user.name}"> <age>${user.age}</age> </user> </root> ``` 然后,我们可以编写 Java 代码,使用 Freemarker 解析模板文件,并将所需数据填充到模板中,生成 XML 文件。 示例 Java 代码: ```java import freemarker.template.Configuration; import freemarker.template.Template; import freemarker.template.TemplateException; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.util.HashMap; import java.util.Map; public class FreemarkerXmlGenerator { public static void main(String[] args) throws IOException, TemplateException { // 创建 Configuration 实例 Configuration configuration = new Configuration(Configuration.getVersion()); // 设置模板文件所在目录 configuration.setDirectoryForTemplateLoading(new File("/path/to/template/dir")); // 加载模板文件 Template template = configuration.getTemplate("template.ftl"); // 准备数据 Map<String, Object> data = new HashMap<>(); data.put("user", new User(1, "Tom", 20)); // 创建输出文件 File outputFile = new File("/path/to/output/file.xml"); FileWriter fileWriter = new FileWriter(outputFile); // 填充数据并生成 XML 文件 template.process(data, fileWriter); // 关闭输出流 fileWriter.close(); } private static class User { private int id; private String name; private int age; public User(int id, String name, int age) { this.id = id; this.name = name; this.age = age; } public int getId() { return id; } public String getName() { return name; } public int getAge() { return age; } } } ``` 以上代码中,我们通过 `Configuration` 类加载模板文件,然后用 `Template` 类进行解析和填充,并将结果输出到指定的文件中。注意要在最后关闭输出流。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 67
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值