文档章节

java生成word(文字和图片)

RampageAngle
 RampageAngle
发布于 2016/07/15 12:02
字数 1024
阅读 66
收藏 0
点赞 0
评论 0

1、整体思路
    利用xml模板,在模板中预留占位标识(${yourContent}),然后将xml转为ftl文件,通过Map<String, Object>传值填充对应的内容即可,word其实和html一样,也有自己的xml标签,表头、段落、图片、以及字体、标题等的标签。文字必须包含在段落中,如:<w:p><w:r><w:t>${yourContent}</w:t></w:r></w:p>,图片必须是在
<w:pict><w:binData>${image}<w:binData><v:shape><v:imagedata</v:shape>
其中<w:binData>、<v:shape>和<v:imagedata>有几个必填参数,后面的代码会涉及到。集合的循环遍历通过<#list yourListName as alias>${alias.property}...</#list>方式实现。
2、生成模板
    新建word模板,设置自己要替换的内容。比如下面例子:
        我的文档
        作者:author 时间:time 内容:content
        把新建好的word导出成word2003xml文件,然后将author改成${author},time改成${time},content改成${content}然后修改后缀名为ftl文件,存到项目里面。
3、替换内容
    实际操作中,不光会遇到纯文字的,经常会遇到文字加图片。图片的实现:先转换成base64的字符串,然后填充到图片的标签中(网络图片,必须先下载到本地才能转换)。下面是转换的代码:
----------

/**
     * 替换内容中的图片,样式
     * 
     * @param content
     * @return
     * @throws Exception
     */
    public String replaceImage(String content) throws Exception {
        try {
            Pattern p = Pattern.compile("<img[^>]+src\\s*=\\s*['\"]([^'\"]+)['\"][^>]*>");
            Matcher m = p.matcher(content);
            int i = 0;
            content = StyleFilter.shieldStyle(content);
            while (m.find()) {
                String width = getWidth(m.group());
                String height = getHeight(m.group());
                String target = getWholeImage(m.group(1), width, height);

                content = content.replace(m.group(), target);
            }
        } catch (ConfigurationException e) {
            e.printStackTrace();
        }
        return content;
    }
    
    public String getWidth(String content) {
        String regex = "width=['\"]?(.*?)['\"]?\\s.*?>";

        Pattern p = Pattern.compile(regex);
        Matcher m = p.matcher(content);
        while (m.find()) {
            return m.group(1);
        }
        return null;
    }

    public String getHeight(String content) {
        String regex = "height=['\"]?(.*?)['\"]?\\s.*?>";

        Pattern p = Pattern.compile(regex);
        Matcher m = p.matcher(content);
        while (m.find()) {
            return m.group(1);
        }
        return null;
    }
    
    /**
     * 生成图片(独立成段落的图片,仅前半部分,后半部分手动拼接)
     * 
     * @param imgUrl
     * @param width
     * @param height
     * @return
     * @throws Exception
     */
    public String getWholeImage(String imgUrl, String width, String height) throws Exception {
        String result = "";
        try {
            if (null != width && null != height) {
                String no = Math.random() * 100 + "";
                String binData1 = "<w:pict><w:binData w:name=\"wordml://" + no + ".png\" xml:space=\"preserve\">";
                String binData2 = "</w:binData><v:shape id=\"" + no + "\" type=\"#_x0000_t75\" style=\"width:" + width
                        + ";height:" + height + "\"><v:imagedata src=\"wordml://" + no + ".png\"/></v:shape>";
                result = binData1 + getImageStr(imgUrl) + binData2 + "</w:pict>";
            } else {
                String no = Math.random() * 100 + "";
                String binData1 = "<w:pict><w:binData w:name=\"wordml://" + no + ".png\" xml:space=\"preserve\">";
                String binData2 = "</w:binData><v:shape id=\"" + no
                        + "\" type=\"#_x0000_t75\"><v:imagedata src=\"wordml://" + no + ".png\"/></v:shape>";
                result = binData1 + getImageStr(imgUrl) + binData2 + "</w:pict>";
            }
        } catch (ConfigurationException e) {
            e.printStackTrace();
        }
        return result;
    }
其中这两块的名字必须是一致的![](leanote://file/getImage?fileId=57885a94128c7b03ba000000)
/**
     * 图片转码
     * 
     * @return 返回图片base64字符串
     * @throws Exception
     */
    public static String getImageStr(String imgUrl) throws Exception {
        String imgPath = download(imgUrl, Math.random() * 100 + ".png", Config.ReadStringPropertie("pictureDownload"));
        InputStream in = null;
        byte[] data = null;
        try {
            in = new FileInputStream(imgPath);
            
            data = new byte[in.available()];
            in.read(data);
            in.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
        BASE64Encoder encoder = new BASE64Encoder();
        return encoder.encode(data);
    }

/**
     * 下载图片的方法(图片下载到本地后才可转化成base64字符串)
     * 
     * @param urlString
     * @param fileName
     * @param savePath
     * @return
     * @throws Exception
     */
    public static String download(String urlString, String fileName, String savePath) throws Exception {
        // 构造URL
        URL url = new URL(urlString);
        // 打开连接
        URLConnection con = url.openConnection();
        // 设置请求超时为5s
        con.setConnectTimeout(5 * 1000);
        // 输入流
        InputStream is = con.getInputStream();

        // 1K的数据缓冲
        byte[] bs = new byte[1024];
        // 读取到的数据长度
        int len;
        // 输出的文件流
        File sf = new File(savePath);
        if (!sf.exists()) {
            sf.mkdirs();
        }
        String imgUrl = sf.getPath() + "\\" + fileName;
        OutputStream os = new FileOutputStream(sf.getPath() + "\\" + fileName);
        // 开始读取
        while ((len = is.read(bs)) != -1) {
            os.write(bs, 0, len);
        }
        // 完毕,关闭所有链接
        os.close();
        is.close();
        return imgUrl;
    }
    
    enter code here
    调用replaceImage方法获取处理过的字符串,然后赋值到Map中,再调用创建word的方法:
    public String createWord(Map<String, Object> dataMap) {
        String path = "/myWrongBook" + Math.random() * 3 + ".doc";
        try {
            Template t = null;
            configuration.setClassForTemplateLoading(this.getClass(), "your template path"); // FTL文件所存在的位置

            t = configuration.getTemplate("template.ftl");

            File outFile = new File(Config.ReadStringPropertie("wrongBooks") + path);
            Writer out = null;
            out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "UTF-8"));

            t.process(dataMap, out);
            out.close();
            path = Config.ReadStringPropertie("serverPath") + "/wrongBooks" + path;
            //Config.ReadStringPropertie("serverPath")是获取服务器路径的方法,就不给出了
        } catch (IOException ie) {
            ie.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return path;
    }
----------
path就是最终生成的word下载地址,不过这样的地址会比较长,给用户体验不是很爽,可以利用新浪云计算短网址转换成短链接http://jump.sinaapp.com/
    enter code here
    

© 著作权归作者所有

共有 人打赏支持
RampageAngle
粉丝 2
博文 4
码字总数 3380
作品 0
兰州
程序员
Apache POI 操作Word部分

Apache POI ----Word部分 最近在研究Apache POI,不过只研究了如何做word的部分。网上对于Excel等的介绍也很多例子也很多,但是对于word缺少的可怜,导致我学的也很费劲,只能算是会了皮毛。...

QH_C
2014/11/15
0
0
聊聊并发(二)Java SE1.6中的Synchronized

在多线程并发编程中Synchronized一直是元老级角色,很多人都会称呼它为重量级锁,但是随着Java SE1.6对Synchronized进行了各种优化之后,有些情况下它并不那么重了,本文详细介绍了Java SE1....

陶邦仁
2015/03/18
0
0
使用 Grapicmagick 和 Im4java 处理图片

ImageMagick是个图片处理工具可以安装在绝大多数的平台上使用,Linux、Mac、Windows都没有问题。GraphicsMagick是在ImageMagick基础上的另一个项目,大大提高了图片处理的性能,在linux平台上...

voole
05/07
0
0
ios程序本地化---nib版本

一、简介 * 使用本地化功能,可以轻松地将应用程序翻译成多种语言,甚至可以翻译成同一语言的多种方言 * 如果要添加本地化功能,需要为每种支持的语言创建一个子目录,称为”本地化文件夹”,...

孔俊
2013/06/19
0
0
使用freemarker生成word文档

项目有时候需要将一些内容导出成word格式,实现方式很多种,如:POI导出,freemarker导出。freemarker导出比较简单。 主要分三步: 新建一个word文档 生成模板 动态生成word。 新建一个word文...

leesama
2014/05/01
0
3
没有安装Microsoft Words的情况下读写Word文档的控件Aspose.Words​

Aspose.Words控件是一个类库,使您可以执行大部分文件处理的工作。 Aspose.Words支持DOC,OOXML,RTF,HTML,OpenDocument,PDF,XPS,EPUB和其他格式。采用Aspose.Words您可以生成,修改,转...

netkongjian
2014/06/21
0
0
SpannableString与SpannableStringBuilder

一、概述 1、SpannableString、SpannableStringBuilder与String的关系 首先SpannableString、SpannableStringBuilder基本上与String差不多,也是用来存储字符串,但它们俩的特殊就在于有一个...

菜南瓜
2015/05/18
0
0
资源分享-开发工具-Java2ObjC

本文来自我我的博客:https://www.lwxshow.com/soft/resource-sharing-tools-java2objc.html 今天为大家推荐一款:java转objective-c代码工具. 大家都想直接将java代码转成objective-c代码,手...

卢卫湘
2013/06/17
0
0
仿百度文库word文档在线阅读

第一种实现方案:openoffice+swftools+flexpaper 用户上传一个文档,格式有可能是doc、docx、xls、ppt、pdf 等。其他用户在前台查看的却是一个swf格式的flash文档。 flash文档可以起到防止用...

JellyFlower
2014/01/18
0
1
Android项目实战(一): SpannableString与SpannableStringBuilder(转)

前言: 曾经在一些APP中的一些类似“帮助”“关于”的界面看过一行文字显示不同的颜色的效果,如下效果: 本软件是一款高、大、上的社区类软件。 一般来说,这应该是由一个TextView来显示的,...

听着music睡
2014/11/13
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

SpringBoot | 第十章:Swagger2的集成和使用

前言 前一章节介绍了mybatisPlus的集成和简单使用,本章节开始接着上一章节的用户表,进行Swagger2的集成。现在都奉行前后端分离开发和微服务大行其道,分微服务及前后端分离后,前后端开发的...

oKong
今天
5
0
Python 最小二乘法 拟合 二次曲线

Python 二次拟合 随机生成数据,并且加上噪声干扰 构造需要拟合的函数形式,使用最小二乘法进行拟合 输出拟合后的参数 将拟合后的函数与原始数据绘图后进行对比 import numpy as npimport...

阿豪boy
今天
1
0
云拿 无人便利店

附近(上海市-航南路)开了家无人便利店.特意进去体验了一下.下面把自己看到的跟大家分享下. 经得现场工作人员同意后拍了几张照片.从外面看是这样.店门口的指导里强调:不要一次扫码多个人进入....

周翔
昨天
1
0
Java设计模式学习之工厂模式

在Java(或者叫做面向对象语言)的世界中,工厂模式被广泛应用于项目中,也许你并没有听说过,不过也许你已经在使用了。 简单来说,工厂模式的出现源于增加程序序的可扩展性,降低耦合度。之...

路小磊
昨天
165
1
npm profile 新功能介绍

转载地址 npm profile 新功能介绍 npm新版本新推来一个功能,npm profile,这个可以更改自己简介信息的命令,以后可以不用去登录网站来修改自己的简介了 具体的这个功能的支持大概是在6这个版...

durban
昨天
1
0
Serial2Ethernet Bi-redirection

Serial Tool Serial Tool is a utility for developing serial communications, custom protocols or device testing. You can set up bytes to send accordingly to your protocol and save......

zungyiu
昨天
1
0
python里求解物理学上的双弹簧质能系统

物理的模型如下: 在这个系统里有两个物体,它们的质量分别是m1和m2,被两个弹簧连接在一起,伸缩系统为k1和k2,左端固定。假定没有外力时,两个弹簧的长度为L1和L2。 由于两物体有重力,那么...

wangxuwei
昨天
0
0
apolloxlua 介绍

##项目介绍 apolloxlua 目前支持javascript到lua的翻译。可以在openresty和luajit里使用。这个工具分为两种模式, 一种是web模式,可以通过网页使用。另外一种是tool模式, 通常作为大规模翻...

钟元OSS
昨天
2
0
Mybatis入门

简介: 定义:Mybatis是一个支持普通SQL查询、存储过程和高级映射的持久层框架。 途径:MyBatis通过XML文件或者注解的形式配置映射,实现数据库查询。 特性:动态SQL语句。 文件结构:Mybat...

霍淇滨
昨天
2
0
开发技术瓶颈期,如何突破

前言 读书、学习的那些事情,以前我也陆续叨叨了不少,但总觉得 “学习方法” 就是一个永远在路上的话题。个人的能力、经验积累与习惯方法不尽相同,而且一篇文章甚至一本书都很难将学习方法...

_小迷糊
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部