使用Java生成word文档(附源码)

简介:

 当我们使用Java生成word文档时,通常首先会想到iTextPOI,这是因为我们习惯了使用这两种方法操作Excel,自然而然的也想使用这种生成word文档。但是当我们需要动态生成word时,通常不仅要能够显示word中的内容,还要能够很好的保持word中的复杂样式。这时如果再使用ITextPOI去操作,就好比程序员去搬砖一样痛苦。


  这时候,我们应该考虑使用FreeMarker的模板技术快速实现这个复杂的功能,让程序员在喝咖啡的过程中就把问题解决。实现思路是这样的:先创建一个word文档,按照需求在word中填好一个模板,然后把对应的数据换成变量${},然后将文档保存为xml文档格式,使用文档编辑器打开这个xml格式的文档,去掉多余的xml符号,使用Freemarker读取这个文档然后替换掉变量,输出word文档即可。

  具体过程如下:

  1.创建带有格式的word文档,将该需要动态展示的数据使用变量符替换。

140726908.png

2. 将刚刚创建的word文档另存为xml格式。

140752129.png


3编辑这个XMl文档去掉多余的xml标记,如图中蓝色部分

140826202.png



  4.从Freemarker官网【下载】最新的开发包,将freemarker.jar拷贝到自己的开发项目中。

  5.新建DocUtil类,实现根据Doc模板生成word文件的方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
package  com.favccxx.secret.util;
import  java.io.BufferedWriter;
import  java.io.File;
import  java.io.FileOutputStream;
import  java.io.OutputStreamWriter;
import  java.io.Writer;
import  java.util.Map;
import  freemarker.template.Configuration;
import  freemarker.template.DefaultObjectWrapper;
import  freemarker.template.Template;
import  freemarker.template.TemplateExceptionHandler;
public  class  DocUtil {
        privateConfiguration configure =  null ;
        publicDocUtil(){
               configure=  new  Configuration();
               configure.setDefaultEncoding( "utf-8" );
        }
        /**
         * 根据Doc模板生成word文件
         * @param dataMap Map 需要填入模板的数据
         * @param fileName 文件名称
         * @param savePath 保存路径
         */
        publicvoid createDoc(Map<String, Object> dataMap, String downloadType, StringsavePath){
               try {
                      //加载需要装填的模板
                      Templatetemplate  =  null ;
                      //加载模板文件
                      configure.setClassForTemplateLoading( this .getClass(), "/com/favccxx/secret/templates" );
                      //设置对象包装器
                      configure.setObjectWrapper(newDefaultObjectWrapper());
                      //设置异常处理器
                      configure.setTemplateExceptionHandler(TemplateExceptionHandler.IGNORE_HANDLER);
                      //定义Template对象,注意模板类型名字与downloadType要一致
                      template= configure.getTemplate(downloadType +  ".xml" );
                      //输出文档
                      FileoutFile =  new  File(savePath);
                      Writerout =  null ;
                      out=  new  BufferedWriter( new  OutputStreamWriter( new  FileOutputStream(outFile), "utf-8" ));                                   
                      template.process(dataMap,out);
                      outFile.delete();
               } catch  (Exception e) {
                      e.printStackTrace();
               }
        }
}

  6.用户根据自己的需要,调用使用getDataMap获取需要传递的变量,然后调用createDoc方法生成所需要的文档。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
  * 根据下载类型获取需要传递的Map参数
  * @param oid 对象Id
  * @param downloadType 下载类型
  */
private  Map<String, Object> getDataMap(String oid,String downloadType){
     Map<String, Object> dataMap =  new  HashMap<String, Object>();
     if ( "Parameter1" .equals(downloadType)){
         ...
         ...
         dataMap = DataMapUtil.setObjToMap(Object1);
     } else {
         ...
         ...
         dataMap = DataMapUtil.setObjToMap(Object2);
     }
     return  dataMap;
}

  

  

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
public  class  DataMapUtil {
       
     private  static  Map<String, Object> dataMap =  new  HashMap<String, Object>();
       
     /**
      * 将对象转换成Map
      * @param obj 对象类
      * @return
      */
     public  static  Map<String,Object> setObjToMap(Object obj){
         Class c;
         try  {
             c = Class.forName(obj.getClass().getName());
             Method[] methods = c.getMethods();
             for ( int  i= 0 ,l=methods.length;i<l;i++){
                 String method = methods[i].getName();
                 System.out.println( "The method is:"  + method);
                 if (method.startsWith( "get" )){
                     Object value = methods[i].invoke(obj);
                     if (value !=  null ){
                         if (value.getClass().getClassLoader() !=  null ){   //处理自定义的对象类型
                             setObjToMap(value);
                         }
                         String key = method.substring( 3 );
                         key = key.substring( 0 1 ).toLowerCase() + key.substring( 1 );
                         if ( "java.util.Date" .equals(value.getClass().getName())){
                             value = DateUtil.dateToString((Date)value);
                         }
                         dataMap.put(key, value);
                     }
                 }
             }
         catch  (Exception e) {
             e.printStackTrace();
         }
         return  dataMap;
     }
}


  7.赶紧把这个方法,应用到自己的项目中吧。





本文转自 genuinecx 51CTO博客,原文链接:http://blog.51cto.com/favccxx/1331115,如需转载请自行联系原作者

目录
相关文章
|
1天前
|
JavaScript Java 测试技术
基于Java的快递信息管理系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的快递信息管理系统的设计与实现(源码+lw+部署文档+讲解等)
21 5
|
1天前
|
JavaScript Java 测试技术
基于Java的通讯录管理系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的通讯录管理系统的设计与实现(源码+lw+部署文档+讲解等)
19 5
|
1天前
|
JavaScript Java 测试技术
基于Java的普通话培训信息管理系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的普通话培训信息管理系统的设计与实现(源码+lw+部署文档+讲解等)
23 9
|
1天前
|
JavaScript Java 测试技术
基于Java的在线开放课程的设计与实现(源码+lw+部署文档+讲解等)
基于Java的在线开放课程的设计与实现(源码+lw+部署文档+讲解等)
24 3
|
1天前
|
JavaScript Java 测试技术
基于Java的个人消费管理系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的个人消费管理系统的设计与实现(源码+lw+部署文档+讲解等)
16 4
|
1天前
|
JavaScript Java 测试技术
基于Java的大学生校园招聘网的设计与实现(源码+lw+部署文档+讲解等)
基于Java的大学生校园招聘网的设计与实现(源码+lw+部署文档+讲解等)
12 3
|
1天前
|
JavaScript Java 测试技术
基于Java的社区人员管理系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的社区人员管理系统的设计与实现(源码+lw+部署文档+讲解等)
10 2
|
8天前
|
安全 算法 Java
深入理解Java并发编程:线程安全与性能优化
【4月更文挑战第11天】 在Java中,高效的并发编程是提升应用性能和响应能力的关键。本文将探讨Java并发的核心概念,包括线程安全、锁机制、线程池以及并发集合等,同时提供实用的编程技巧和最佳实践,帮助开发者在保证线程安全的前提下,优化程序性能。我们将通过分析常见的并发问题,如竞态条件、死锁,以及如何利用现代Java并发工具来避免这些问题,从而构建更加健壮和高效的多线程应用程序。
|
1天前
|
安全 Java
java多线程(一)(火车售票)
java多线程(一)(火车售票)
|
1天前
|
安全 Java 调度
Java并发编程:深入理解线程与锁
【4月更文挑战第18天】本文探讨了Java中的线程和锁机制,包括线程的创建(通过Thread类、Runnable接口或Callable/Future)及其生命周期。Java提供多种锁机制,如`synchronized`关键字、ReentrantLock和ReadWriteLock,以确保并发访问共享资源的安全。此外,文章还介绍了高级并发工具,如Semaphore(控制并发线程数)、CountDownLatch(线程间等待)和CyclicBarrier(同步多个线程)。掌握这些知识对于编写高效、正确的并发程序至关重要。