PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】
版权声明:本文为HaiyuKing原创文章,转载请注明出处!
前言
使用Poi实现android中根据模板文件生成Word文档的功能。这里的模板文件是doc文件。如果模板文件是docx文件的话,请阅读下一篇文章《PoiDocxDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0),目前只能java生成】》
注意:
- POI 4.0.0 is the first release to require Java 8 or newer.
前期准备
1、下载poi相关jar包
官网下载地址:http://poi.apache.org/download.html
如果windows系统,则下载zip文件;如果是linux系统则选择.tar.gz。

将下载后的压缩包解压,会得到以下文件。

| 文件(夹)名 | 作用 |
| docs | 文档(包括API文档和如何使用及版本信息) |
| lib | doc功能实现依赖的包 |
| ooxml-lib | docx功能实现依赖的包 |
| LICENSE | |
| NOTICE | |
| poi-4.0.0.jar | 基础类 |
| poi-examples-4.0.0.jar | 不明确,不知道什么作用 |
| poi-excelant-4.0.0.jar | excel功能实现 |
| poi-ooxml-4.0.0.jar | docx功能实现 |
| poi-ooxml-schemas-4.0.0.jar | docx功能实现相关类 |
| poi-scratchpad-4.0.0.jar | doc功能实现 |
2、制作docx模板或者doc模板文件

代码分析
1、可以看到doc和docx文件分别对应着组件HWPF和XWPF,而HWPF和XWPF则对应着poi-scratchpad和poi-ooxml。

官网地址:http://poi.apache.org/components/index.html
使用步骤
一、项目组织结构图


注意事项:
1、 导入类文件后需要change包名以及重新import R文件路径
2、 Values目录下的文件(strings.xml、dimens.xml、colors.xml等),如果项目中存在,则复制里面的内容,不要整个覆盖
二、导入步骤
1、将poi相关jar文件导入项目中(Demo采用的是module方式)
引用jar文件参考《【Android Studio安装部署系列】十七、Android studio引用第三方库、jar、so、arr文件》
注意:
解析doc文件,需要引用下面的jar文件:
- poi-4.0.0.jar
- poi-scratchpad-4.0.0.jar
- libs目录下的commons-collections4-4.2.jar

2、在poilib和app的build.gradle文件中添加以下代码
poilib
apply plugin: 'com.android.library'
android {
compileSdkVersion 28
defaultConfig {
minSdkVersion 16
targetSdkVersion 28
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
//poi
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'com.android.support:appcompat-v7:28.0.0'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
//Poi=doc
api files('libs/poi-4.0.0.jar')
api files('libs/poi-scratchpad-4.0.0.jar')
api files('libs/commons-collections4-4.2.jar')
}
app
apply plugin: 'com.android.application'
android {
compileSdkVersion 28
defaultConfig {
applicationId "com.why.project.poidemo"
minSdkVersion 16
targetSdkVersion 28
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
//poi
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation 'com.android.support:appcompat-v7:28.0.0'
implementation 'com.android.support.constraint:constraint-layout:1.1.3'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
//poi
implementation project(':poilib')
}
3、在poilib这个module中添加PoiUtils.java文件

4、将模板文件复制到项目的assets目录下

三、使用方法
1、根据doc模板生成doc文件的关键代码
MainActivity.java
package com.why.project.poidemo; import android.content.Context;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View; import com.why.project.poilib.PoiUtils; import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map; public class MainActivity extends AppCompatActivity { private Context mContext; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); mContext = this; //利用doc模板生成doc文件
findViewById(R.id.btn_poi_doc).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
try {
InputStream templetDocStream = getAssets().open("请假单模板2.doc"); String targetDocPath = mContext.getExternalFilesDir("poi").getPath() + File.separator + "请假单2.doc";//这个目录,不需要申请存储权限 Map<String, String> dataMap = new HashMap<String, String>();
dataMap.put("$writeDate$", "2018年10月14日");
dataMap.put("$name$", "HaiyuKing");
dataMap.put("$dept$", "移动开发组");
dataMap.put("$leaveType$", "☑倒休 √年假 ✔事假 ☐病假 ☐婚假 ☐产假 ☐其他");
dataMap.put("$leaveReason$", "倒休一天。");
dataMap.put("$leaveStartDate$", "2018年10月14日上午");
dataMap.put("$leaveEndDate$", "2018年10月14日下午");
dataMap.put("$leaveDay$", "1");
dataMap.put("$leaveLeader$", "同意");
dataMap.put("$leaveDeptLeaderImg$", "同意!"); PoiUtils.writeToDoc(templetDocStream,targetDocPath,dataMap); } catch (IOException e) {
e.printStackTrace();
} }
});
}
}
PoiUtils.java
package com.why.project.poidemo.poi; import org.apache.poi.hwpf.HWPFDocument;
import org.apache.poi.hwpf.usermodel.Range; import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map; /**
* Created by HaiyuKing
* Used poi工具类封装
* 在使用POI写word doc文件的时候我们必须要先有一个doc文件才行,因为我们在写doc文件的时候是通过HWPFDocument来写的,
* 而HWPFDocument是要依附于一个doc文件的。所以通常的做法是我们先在硬盘上准备好一个内容空白的doc文件,然后建立一个基于该空白文件的HWPFDocument。
* 之后我们就可以往HWPFDocument里面新增内容了,然后再把它写入到另外一个doc文件中,这样就相当于我们使用POI生成了word doc文件。
*/
public class PoiUtils { /**
* 生成一个doc文件
* @param templetDocPath 模板文件的完整路径
* @param targetDocPath 生成的目标文件的完整路径
* @param dataMap 替换的数据*/
public static void writeToDoc(String templetDocPath, String targetDocPath, Map<String,String> dataMap){
try
{
//得到模板doc文件的HWPFDocument对象
InputStream in = new FileInputStream(templetDocPath);
writeToDoc(in,targetDocPath,dataMap);
}
catch(IOException e)
{
e.printStackTrace();
}
} /**
* 生成一个doc文件,主要用于直接读取asset目录下的模板文件,不用先复制到sd卡中
* @param templetDocInStream 模板文件的InputStream
* @param targetDocPath 生成的目标文件的完整路径
* @param dataMap 替换的数据*/
public static void writeToDoc(InputStream templetDocInStream, String targetDocPath, Map<String,String> dataMap){
try
{
//得到模板doc文件的HWPFDocument对象
HWPFDocument HDoc = new HWPFDocument(templetDocInStream);
// 获取word文本内容,整个文本
Range range = HDoc.getRange();
// 替换文本内容,将自定义的$xxx$替换成实际文本
for(Map.Entry<String, String> entry : dataMap.entrySet())
{
range.replaceText(entry.getKey(), entry.getValue());
}
//写到另一个文件中
FileOutputStream out = new FileOutputStream(targetDocPath, true);
//把doc输出到输出流中
HDoc.write(out);
out.close();
templetDocInStream.close();
}
catch(IOException e)
{
e.printStackTrace();
}
catch(Exception e)
{
e.printStackTrace();
}
} }
2、效果
生成的文件路径:内部存储——Android——data——com.why.project.poidemo——files——poi——请假单2.doc

混淆配置
无
参考资料
Android使用ApachePOI组件读写Worddoc和docx文件
项目demo下载地址
https://github.com/haiyuKing/PoiDemo
PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】的更多相关文章
- PoiDocxDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0),目前只能java生成】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 这个是<PoiDemo[Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)]>的扩展,上一篇是根 ...
- FreemarkerJavaDemo【Android将表单数据生成Word文档的方案之一(基于freemarker2.3.28,只能java生成)】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 这个方案只能在java中运行,无法在Android项目中运行.所以此方案是:APP将表单数据发送给后台,后台通过freemarker ...
- Android根据word模板文档将表单数据生成word文档的方案整理
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 尝试的方案包括以下几种: freemarker 只能在java项目上运行,无法在Android项目上运行: 参考资料:<Fre ...
- Java Web项目中使用Freemarker生成Word文档
Web项目中生成Word文档的操作屡见不鲜.基于Java的解决方式也是非常多的,包含使用Jacob.Apache POI.Java2Word.iText等各种方式,事实上在从Office 2003開始 ...
- Java 导出数据库表信息生成Word文档
一.前言 最近看见朋友写了一个导出数据库生成word文档的业务,感觉很有意思,研究了一下,这里也拿出来与大家分享一波~ 先来看看生成的word文档效果吧 下面我们也来一起简单的实现吧 二.Java 导 ...
- PowerDesigner将PDM导出生成WORD文档
PowerDesigner将PDM导出生成WORD文档 环境 PowerDesigner15 1.点击Report Temlates 制作模板 2.如果没有模板,单击New图标创建.有直接双击进入. ...
- 将HTML导出生成word文档
前言: 项目开发中遇到了需要将HTML页面的内容导出为一个word文档,所以有了这边随笔. 当然,项目开发又时间有点紧迫,第一时间想到的是用插件,所以百度了下.下面就介绍两个导出word文档的方法. ...
- POI生成word文档完整案例及讲解
一,网上的API讲解 其实POI的生成Word文档的规则就是先把获取到的数据转成xml格式的数据,然后通过xpath解析表单式的应用取值,判断等等,然后在把取到的值放到word文档中,最后在输出来. ...
- 使用freemarker模板引擎生成word文档的开发步骤
1.准备模板文档,如果word文档中有表格,只保留表头和第一行数据:2.定义变量,将word文档中的变量用${var_name}替换:3.生成xml文件,将替换变量符后的word文档另存为xml文件: ...
随机推荐
- java--接口的定义与实现
利用接口方法计算矩形面积 代码如下: //接口的定义与实现[public] interface A{ //定义一个接口[public] [static][final] void conter(doub ...
- 在Java中谈尾递归--尾递归和垃圾回收的比较(转载)
我不是故意在JAVA中谈尾递归的,因为在JAVA中谈尾递归真的是要绕好几个弯,只是我确实只有JAVA学得比较好,虽然确实C是在学校学过还考了90+,真学得没自学的JAVA好 不过也是因为要绕几个弯,所 ...
- 第二课:Hadoop集群环境配置
一.Yum配置 1.检查Yum是否安装 rpm -qa|grep yum 2.修改yum源,我使用的是163的镜像源(http://mirrors.163.com/),根据自己的系统选择源, #进入目 ...
- 整理Cocos2d-x 面试题解
昨天听了腾讯2015校招的在线宣讲会,看到了游戏技术大拿Steven,他总结了需要的达人得爱技术,能坚持,够挑剔.马上校招了,加油吧,骚年~ 网上关于cocos2d-x的面试题比较少,这里搜集和整理了 ...
- __new__()方法的使用和实例化
Python中__new__()方法的使用和实例化 1 2 new()是在新式类中新出现的方法,它作用在构造方法init()建造实例之前,可以这么理解,在Python 中存在于类里面的构造方法init ...
- ubuntu 命令整合2
通配符 * 匹配任意多个字符 ?匹配一个任意字符 示例:ls *.txt rm -rf *.txt 文本编辑器 vi.vim 格式:vi 文件名 编辑 vi的三种工作模式 正常模式(启动进入的模式) ...
- Slim 文档-First Application 翻译
最近刚好在用Slim框架,看文档的时候,中文文档中缺了这一篇没有翻译,于是我就把它翻译过来了.本人英语水平有限,如有错误请告知. 如果你在寻找创建一款非常简单的 Slim 应用程序的流程,来这里算是找 ...
- Linux之SSH密钥认证
1.SSH协议的认识 SSH 为 Secure Shell 的缩写,由 IETF 的网络小组(Network Working Group)所制定:SSH 为建立在应用层基础上的安全协议.SSH 是目前 ...
- java多线程对CountDownLatch的使用实例
介绍 CountDownLatch是一个同步辅助类,它允许一个或多个线程一直等待直到其他线程执行完毕才开始执行. 用给定的计数初始化CountDownLatch,其含义是要被等待执行完的线程个数. 每 ...
- HTTP协议GET HEAD简单介绍
一.HTTP协议简介 超文本传输协议(Hypertext Transfer Protocol,简称HTTP)是应用层协议,自 1990 年起,HTTP 就已经被应用于 WWW 全球信息服务系统. HT ...