FreeMarker生成(导出)Excel表格,提示文件已损坏,无法打开错误解决

最近公司在使用 FreeMarker 生成excel表格导出数据时,使用 office2010 打开时提示,文件已损坏,无法打开,用wps打开是正常的,最后比较文件发现有一列 ss:ExpandedRowCount=“15”,和别的文档不一样(实际列数比这个模板里面的列数多),最后删除导出模板里面的这列后,再用 office2010 成功打开了文档。
在这里插入图片描述

这是因为Excel在生成xml的时候,可能是为了不浪费资源吧,在生成时候,会指定Excel有多少行,如果超出了这个长度的话,它就会编译不通过.报错.
在这里插入图片描述
在生成的ftl里面删除 ss:ExpandedRowCount=“xx” 这一列,再生产excel导出就好了

使用freemarker生成word ,并集成struts2 同时生成及下载文档 资料附有Java源代码和自己总结的使用说明及注意事项 大至预览如下: 1、用word编辑好模板 普通字符串替换为 ${string} 表格循环用标签 姓名:${user.userName} , 性别:${user.sex} 2、将word模板另存为xml格式 3、将xml模板文件后缀名改为.ftl 4、编辑ftl文件 注意 编辑word模板时,${string} 标签最好是手动一次性输入完毕,或者使用记事本统一将整个${string}编辑好之后,粘贴至word里边。 也就是说,不要在word里首先打完 ${ } 之后,又从其它地方把 string 字符串粘贴至 { } 之间,这样在 word 转化为 xml时,解析会有问题,freemarker解析时,会报错。 /** * @Desc:生成word文件 * @Author:张轮 * @Date:2014-1-22下午05:33:42 * @param dataMap word中需要展示的动态数据,用map集合来保存 * @param templateName word模板名称,例如:test.ftl * @param filePath 文件生成的目标路径,例如:D:/wordFile/ * @param fileName 生成文件名称,例如:test.doc */ @SuppressWarnings("unchecked") public static void createWord(Map dataMap,String templateName,String filePath,String fileName){ try { //创建配置实例 Configuration configuration = new Configuration(); //设置编码 configuration.setDefaultEncoding("UTF-8"); //ftl模板文件统一放至 com.lun.template 包下面 configuration.setClassForTemplateLoading(WordUtil.class,"/com/lun/template/"); //获取模板 Template template = configuration.getTemplate(templateName); //输出文件 File outFile = new File(filePath+File.separator+fileName); //如果输出目标文件夹不存在,则创建 if (!outFile.getParentFile().exists()){ outFile.getParentFile().mkdirs(); } //将模板和数据模型合并生成文件 Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile),"UTF-8")); //生成文件 template.process(dataMap, out); //关闭流 out.flush(); out.close(); } catch (Exception e) { e.printStackTrace(); } }
### 解决方案 在 Linux 环境下使用 Freemarker 导出 Excel 文件时,如果下载后的文件显示损坏,通常可能由以下原因导致:编码问题、文件流处理不当、模板生成内容错误或 MIME 类型设置不正确。以下是详细的分析和解决方案。 #### 1. **检查文件编码** 确保生成Excel 文件使用 UTF-8 编码,避免因编码不一致导致文件损坏。Freemarker 模板中可以显式指定字符集: ```ftl <#setting encoding="UTF-8"> ``` 此外,在后端代码中也需要确保输出流的编码与模板一致[^1]。 #### 2. **正确设置 MIME 类型** 在 HTTP 响应头中,必须正确设置 MIME 类型为 `application/vnd.ms-excel` 或 `application/octet-stream`,以确保浏览器能够正确识别并下载文件。示例如下: ```java response.setContentType("application/vnd.ms-excel"); response.setHeader("Content-Disposition", "attachment; filename=exported_file.xlsx"); ``` #### 3. **确保文件流处理完整** 文件流的处理需要特别注意,确保所有数据都正确写入输出流,并且在完成操作后关闭流。以下是典型的代码实现: ```java OutputStream outputStream = response.getOutputStream(); try { // 将生成的内容写入输出流 workbook.write(outputStream); } finally { if (outputStream != null) { outputStream.flush(); outputStream.close(); } } ``` 如果流未正确关闭或数据未完全写入,可能导致文件损坏。 #### 4. **检查模板生成的内容** 确保 Freemarker 模板生成的内容符合 Excel 的格式要求。如果生成的内容包含非法字符或格式错误,也可能导致文件损坏。建议使用 Apache POI 或其他专门的库来生成 Excel 文件,而不是直接通过纯文本方式生成。 #### 5. **验证文件完整性** 在生成文件后,可以在服务器端保存一份临时文件并手动打开验证其完整性。如果文件在服务器端是正常的,但在客户端显示损坏,则可能是网络传输或 MIME 类型设置的问题。 #### 6. **解决跨平台兼容性问题** Linux 环境下的换行符(`\n`)与 Windows 环境下的换行符(`\r\n`)不同,可能会导致文件在某些情况下无法正常解析。可以通过以下方式统一换行符: ```java String content = freemarkerTemplate.process(dataModel); content = content.replace("\n", "\r\n"); ``` --- ### 示例代码 以下是一个完整的示例,展示如何在 Linux 环境下使用 Freemarker 和 Apache POI 导出 Excel 文件: ```java import org.apache.poi.ss.usermodel.*; import org.apache.poi.xssf.usermodel.XSSFWorkbook; import java.io.OutputStream; import java.util.Map; public class ExcelExporter { public void exportExcel(Map<String, Object> dataModel, OutputStream outputStream) throws Exception { // 创建工作簿 Workbook workbook = new XSSFWorkbook(); // 创建工作表 Sheet sheet = workbook.createSheet("Sheet1"); // 使用 Freemarker 模板生成内容 String templateContent = processFreemarkerTemplate(dataModel); // 将生成的内容写入工作表 Row row = sheet.createRow(0); Cell cell = row.createCell(0); cell.setCellValue(templateContent); // 输出到流 try { workbook.write(outputStream); } finally { workbook.close(); if (outputStream != null) { outputStream.flush(); outputStream.close(); } } } private String processFreemarkerTemplate(Map<String, Object> dataModel) throws Exception { // 这里假设已经加载了 Freemarker 模板 // 返回生成的内容 return "Generated Content"; } } ``` --- ###
评论 4
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值