- 浏览: 59088 次
- 性别:
- 来自: 苏州
最新评论
-
se7en8974:
有效。顶一个
老外的lovcombo 使用注意点 -
yekki:
rake rails:freeze:gem与rake rail ...
rake rails:freeze -
tianzhihua:
把你的错误贴上来吧
Struts2 Map 映射 -
myoldman:
我记得java里面integer的最大数值为214748364 ...
Struts2 Map 映射
我们需求如下:给你一个模板,里面有一个表格,标题已经给好,程序主要就是完成把数据填写到word中并提供给用户下载。
网上找了很久,发现主要两种开源包:POI和Jacob,Jacob首先被否决掉了,因为他最后必须运行在windows平台上。Excel导入导出我就是用的POI,但是POI中的word操作实在不行,读取还可以,写入数据远不能满足项目的需要。后面尝试诸如生成PDF然后转Word,生成XML转Word,生成Html转word,生成rtf转word。这些都是可行的办法,但是网上开源包的功能有限,iText对PDF操作是很强,但是对PDF内容的解析不行。
最后这个事情拖了四天,我看文档的时候想到mht文件,于是我把Word模板导出成mht文件,然后查看其源码,其实就是html代码,我想这样可以用velocity填写数据,然后直接导出成doc格式的文档,后来这样基本满足我们项目的需要。
#foreach($bean in $beanList)
<tr style=3D'mso-yfti-irow:1;mso-yfti-lastrow:yes;page-break-inside:avoid;
height:22.7pt'>
<td width=3D"6%" style=3D'width:6.52%;border:solid windowtext 1.0pt;borde=
r-top:
none;mso-border-top-alt:solid windowtext .5pt;mso-border-alt:solid window=
text .5pt;
padding:0cm 5.4pt 0cm 5.4pt;height:22.7pt'>
<p class=3DMsoNormal align=3Dcenter style=3D'text-align:center'><span style=3D'mso-bidi-font-size:10.5pt;font-family:SimSun'>${bean.seqNo}<o:p></o:p></span=
></p>
</td>
<td width=3D"10%" style=3D'width:10.12%;border-top:none;border-left:none;
border-bottom:solid windowtext 1.0pt;border-right:solid windowtext 1.0pt;
mso-border-top-alt:solid windowtext .5pt;mso-border-left-alt:solid window=
text .5pt;
mso-border-alt:solid windowtext .5pt;padding:0cm 5.4pt 0cm 5.4pt;height:2=
2.7pt'>
<p class=3DMsoNormal align=3Dcenter style=3D'text-align:center'><span
style=3D'mso-bidi-font-size:10.5pt;font-family:SimSun'>${bean.name}<span
lang=3DEN-US><o:p></o:p></span></span></p>
</td>
</tr>
#end
上面是用mht源码改写成vm文件的部分代码,就和生成一般的文本文件一摸一样。这儿如果直接把汉字、特殊字符什么的传给模板(当然英文不会),会出现乱码,导致根据模板导出的word文件这个都是乱码,关于乱码的详细解释可以参考:http://blog.csdn.net/myyate/archive/2008/04/08/2260234.aspx ; 也就是说导出时需要把带汉字的字符串都进行转码,这儿给出一些方法:
/**
* 把给定的str转换为10进制表示的unicode,格式为:姨
* 目前只是用于mht模板的转码
* @param str
* @return
*/
public static String encode2HtmlUnicode(String str) {
if(str == null) return "";
StringBuilder sb = new StringBuilder(str.length() * 2);
for (int i = 0; i < str.length(); i++) {
sb.append(encode2HtmlUnicode(str.charAt(i)));
}
return sb.toString();
}
public static String encode2HtmlUnicode(char character) {
if (character > 255) {
return "&#" + (character & 0xffff) + ";";
} else {
return String.valueOf(character);
}
}
public static String encode2HtmlUnicode(Character character) {
if(character == null) return null;
return encode2HtmlUnicode(character.charValue());
}
public static void encode2HtmlUnicode(String[] value) {
if(value == null || value.length < 1) return;
for(int i = 0; i < value.length; i ++) {
value[i] = encode2HtmlUnicode(value[i]);
}
}
上面的这些方法在项目中可以通过递归来对bean进行转码,当然我在项目中只是针对String和Character两种数据类型进行转码。
static Object encodeStringAndCharacter(Object value) {
if(value instanceof String) {
return StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((String) value));
} else if(value instanceof Character) {
return StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((Character) value));
}
return null;
}
static void encodeExceptStringAndCharacter(Object value) throws Exception {
if(value instanceof String[]) {
EncodeUtils.encode2HtmlUnicode((String[]) value);
} else if(value instanceof List) {
encode2HtmlUnicode((List) value);
} else if(value instanceof Form) {
encode2HtmlUnicode((Form) value);
} else if(value instanceof Form[]) {
encode2HtmlUnicode((Form[]) value);
}
}
@SuppressWarnings("unchecked")
static void encode2HtmlUnicode(List value) throws Exception {
if(value == null || value.size() < 1) return;
for(int i = 0; i < value.size(); i ++) {
Object ele = value.get(i);
Object result = encodeStringAndCharacter(ele);
if(result != null) {
value.set(i, result);
} else {
encodeExceptStringAndCharacter(ele);
}
}
}
static void encode2HtmlUnicode(Form[] value) throws Exception {
if(value == null) return;
for(int i = 0; i < value.length; i ++) {
encode2HtmlUnicode(value[i]);
}
}
static void encode2HtmlUnicode(Form value) throws Exception {
if(value == null) return;
PropertyDescriptor[] pds = PropertyUtils.getPropertyDescriptors(value.getClass());
if(pds == null) return;
for(int i = 0; i < pds.length; i ++) {
PropertyDescriptor pd = pds[i];
Object fieldValue = PropertyUtils.getProperty(value, pd.getName());
Class fieldType = pd.getPropertyType();
if(String.class.isAssignableFrom(fieldType)) {
PropertyUtils.setProperty(value,
pd.getName(),
StringUtils.replaceNullString(EncodeUtils.encode2HtmlUnicode((String) fieldValue)));
} else if(List.class.isAssignableFrom(fieldType)) {
encode2HtmlUnicode((List) fieldValue);
} else if(Form.class.isAssignableFrom(fieldType)) {
encode2HtmlUnicode((Form) fieldValue);
} else if(Form[].class.isAssignableFrom(fieldType)) {
encode2HtmlUnicode((Form[]) fieldValue);
}
}
}
上面的Form类型不需要关心,就是一个Bean接口。这样在传到模板之前进行上述转码,就基本上可以避免乱码问题。
导出结束后,只需要把OutputStream导出到*.doc文件中即可,下载的话只要设置:
response.setHeader("Content-disposition", "attachment; filename=" + URLEncoder.encode("***.doc", DEFAULT_WEB_ENCODING));
项目还有一个需求,对于一些飞机零件或者领导名字要求是图片显示。这个不难,但模板做起比较繁琐,因为mht模板和难控制,一般都是通过word另存的,而且里面汉字有时ascii格式,所以需要显示图片的地方都事先放一个图片,然后再在mht代码中替换一下:
#if($manager)
<!--[if gte vml =
1]><v:shape
id=3D"_x0000_i1026" type=3D"#_x0000_t75" style=3D'width:96pt;height:96pt'>
<v:imagedata src=3D"file6086.files/image003.gif" o:title=3D"5080"/>
</v:shape><![endif]--><![if !vml]><img width=3D128 height=3D128
src=3D"file6086.files/image003.gif" v:shapes=3D"_x0000_i1026"><![endif]>
#end
......
#if($manager)
------=_NextPart_01C8E27B.AE1C3C80
Content-Location: file:///C:/C071D706/file6086.files/image003.gif
Content-Transfer-Encoding: base64
Content-Type: image/gif
${manager}
#end
在mht文件中,图片都是以base64码来存储的,所以在存取之前,必须把二进制的图片进行base64编码:
static BASE64Encoder base64Encoder = new BASE64Encoder();
/**
* 把给定的二进制流变成base64码
* @param in
* @return
* @throws IOException
*/
public static synchronized String base64Encode(InputStream in) throws IOException {
byte[] img = new byte[in.available()];
in.read(img);
return base64Encode(img);
}
public static synchronized String base64Encode(String in) throws IOException {
return base64Encode(in.getBytes());
}
public static synchronized String base64Encode(byte[] in) throws IOException {
return base64Encoder.encode(in);
}
转换成String以后在传到模板文件中显示则可。关于任何格式、布局等等等的调整你可以事先在word里先全部调整好再另存。导出效果如下图:
上面的“编号”、表格数据、横线上的都是填写的。
这种导出到word的方法的一个缺点是模板复杂繁琐,尤其修改的时候更是如此;而且必须是officexp及其以上版本。
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/myyate/archive/2008/01/17/2049701.aspx
发表评论
-
Spring Security 3.x 完整入门教程
2010-09-06 10:20 2571Spring Security 3.x 完整入门教程 本Blo ... -
setup.linux
2010-09-01 17:05 966yum -y install gccyum -y instal ... -
Linux下创建与解压tar, tar.gz和tar.bz2文件及压缩率对比
2010-08-30 15:48 4838Linux下创建与解压tar, tar.gz和tar.bz2文 ... -
linux下编译安装php的参数
2010-08-27 13:57 1524linux下编译安装php的参数 Fast-CGI: ... -
MySQL+PHP5+Apache安装手记
2010-08-27 13:45 920http://database.51cto.com/art/2 ... -
安装PHP
2010-08-27 11:05 2908yum install g++ yum insta ... -
安装mysql
2010-08-27 10:38 846先检查系统是否装上了mysql,如果有将其卸载掉rpm -qa ... -
rpm
2010-08-27 10:22 1103rpm -qpl xxxxxx.rpm 1.如何安装rp ... -
linux下编译安装php的参数汇总
2010-08-27 10:08 1585Fast-CGI:./configure --prefix=/ ... -
iptables用法
2010-08-27 09:32 2117iptables -A INPUT -p tcp --dpor ... -
linux的chmod,chown命令 详解
2010-08-26 16:38 888指令名称 : chmod 使用权限 : 所有使用者 使用方式 ... -
Install and Configure MySQL on Linux
2010-08-26 16:35 1569wget http://mirrors.sohu.com/ ... -
mysql
2010-08-26 14:21 7881、数据库目录 /var/lib/mysql/ 2、 ... -
SQL树查询
2010-02-24 17:36 1589引用: --测试数据 深度排序 DEC ... -
基于java的程序OutOfMemory问题的解决及Xms/Xmx/Xss的解释和应用
2010-02-24 16:44 2094长期以来一直都是做java应用的开发,所使用的开发工具基本上也 ... -
neybean支持漂亮的中文
2009-10-13 17:08 0neybean支持漂亮的中文方法如下: -
sychroflow 状态
2009-01-13 18:00 7291. 创建状态 2. 运行状态 3. 挂起 4. 正常 ...
相关推荐
总之,Java 使用 FreeMarker 导出 Word 是一个灵活且强大的解决方案,尤其适用于需要大量动态生成文档的场景。通过熟练掌握 FreeMarker 和相关的 Java 库,你可以创建出满足各种需求的 Word 文档。当然,由于描述中...
Java完美生成word的解决方案 Java完美生成word的解决方案是指使用Java语言生成Word文档的解决方案。该解决方案使用POI和iText库来生成Word文档,但是存在一些问题,如样式控制不便、文档格式变化等。为了解决这些...
综上所述,"java web项目无刷新导出word文档demo"项目利用FreeMarker模板引擎实现了在用户无感知的情况下动态生成Word文档,提供了一种高效且用户体验良好的数据导出解决方案。通过学习和理解这个项目,开发者可以将...
在本项目中,“使用java Apache poi 根据word模板生成word报表.rar”是一个压缩包,其中包含了一套使用Java和Apache POI库来根据预设的Word模板生成定制化报表的解决方案。 首先,我们需要理解Apache POI API的工作...
这个工具的主要目的是为了提供一种高效、完善的解决方案,尤其在处理大量数据时,能够实现高并发下的稳定性能。在标题和描述中提到的几个关键功能,我们将详细探讨如何使用Easy POI进行Excel模板导出、HTML导出为...
### Java带格式导出Word文档方法详解 #### 一、背景与挑战 在日常工作中,我们经常需要将数据导出到Word文档中,以便于打印或进一步编辑。然而,在使用Java进行开发时,如何高效地生成带有复杂样式的Word文档一直...
JavaWord文档导出jar包与Freemarker是一个用于在Java后端生成动态Word文档的解决方案。这个rar压缩包中包含的jar文件可以帮助开发者利用Freemarker模板语言将数据动态地填充到Word文档中,实现自动化文档生成。下面...
在实际应用中,你可能会遇到以下问题和解决方案: - **特殊字符处理**:由于Word和FreeMarker的编码差异,可能需要对特殊字符进行转义,以防止解析错误。 - **图片和复杂格式**:如果模板包含图片或其他复杂格式,...
这个工具以其高效性和灵活性在IT行业中受到广泛应用,尤其对于需要进行大量文档操作的企业级项目,它提供了一种方便的解决方案。 在“使用Aspose.Words for Java完成复杂Word与PDF的导出”这个主题中,我们可以深入...
这个项目提供了一个完整的解决方案,使得开发者无需深入了解Word文档的内部结构,就能方便地在Java中生成包含表格和图片的Word文档。对于那些需要在Java应用中生成复杂Word文档的开发者来说,这是一个非常实用的资源...
本项目“生成WORD freemarker导出WORD”正是这样一个解决方案,它允许开发者使用FreeMarker模板来创建和导出Microsoft Word文件。 首先,我们要理解FreeMarker的工作原理。FreeMarker是一个基于Java的模板语言,它...
压缩包可能包含一份指南,详细说明如何配置和使用提供的Java代码和模板来导出Word文档。这可能涵盖模板的编写规则,数据模型的设计,以及执行导出的步骤。 6. **模板设计**: 在Freemarker中,模板通常包含文本和...
- 使用`PublicExportToWord.java`中的方法,加载模板文件并替换占位符,这可能涉及到解析文本文件(`导出word.txt`)中的数据。 - 如果需要处理表格数据,`AsposeWordOrCell.java`的函数会派上用场,将数据源中的...
本示例“java导出word2”提供了一种无需预先生成模板的解决方案,使得开发者可以直接在代码中动态构建Word文档。下面我们将深入探讨这个主题,了解如何在Java中创建和导出Word文档。 首先,Java本身并不直接支持...
总的来说,这个项目提供了一个完整的解决方案,从读取Word模板,填充数据,到生成和导出新的Word文档。通过学习和应用这个项目,开发者可以更好地掌握Java环境下使用Apache POI处理Word文档的能力,提升自动化文档...
在Java开发中,生成Word文档是一项常见的需求,例如在报告生成、数据导出等场景。RtfTemplate是一个方便的工具库,它可以帮助开发者快速、高效地生成Word文档,特别是基于RTF(Rich Text Format)格式的文件。这个库...
在本项目中,"SpringBootMybatis+poi+Thymeleaf实现excel文件数据导入到数据库以及从数据库将数据导出成excel.zip",我们主要关注的是如何利用Java技术栈来处理Excel文件,并与数据库进行交互。以下是相关知识点的...
在“aspose模板导出word及pdf打印”这个主题中,我们将深入探讨如何利用Aspose来操作模板并导出为Word文档以及打印或转换为PDF。 1. **Aspose.Word API**: Aspose.Word是Aspose产品家族的一员,专门用于处理...
#### 解决方案一:DB数据导出至Word - **流程**: 1. 使用iText创建Word文档实例。 2. 添加封面页和其他初始内容。 3. 按照预定的格式,将数据库表字段及格式作为对象写入Word文档。 #### 解决方案二:HTML数据...