Bootstrap

Java导入导出word文档中的表格

使用poi导出列表数据到word文档中的表格中,表格支持自定义单元格合并,并可选择导出的列。
导入带表格的word文档,支持合并单元格的表格内容读取。
在这里插入图片描述

maven依赖

<!--Java读取word中的表格数据-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-scratchpad</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>fr.opensagres.xdocreport</groupId>
            <artifactId>xdocreport</artifactId>
            <version>1.0.6</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml-schemas</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>ooxml-schemas</artifactId>
            <version>1.3</version>
        </dependency>
        <!--  poi工具类-->
        <dependency>
            <groupId>com.deepoove</groupId>
            <artifactId>poi-tl</artifactId>
            <version>1.10.0</version>
        </dependency>

使用word模板文件

创建word模板文件时注意使用office word创建模板文件
在这里插入图片描述

将模板文件放到resource下

在这里插入图片描述

获取resource文件夹下文件位置

	/**
     * 功能描述:
     * 获取resource目录下的文件位置
     * @Date: 2023-02-03 09:36:35
     * @Param  templateFileName:
     * @return: java.lang.String
     * @since: 1.0.0
     */
    public String getTemplateAbsolutePath(String templateFileName) throws Exception {
    	//注意getResource("")里面是空字符串
        String path = this.getClass().getClassLoader().getResource(templateFileName).getPath();
        System.out.println(path);
        //如果路径中带有中文会被URLEncoder,因此这里需要解码
        String filePath = URLDecoder.decode(path, "UTF-8");
        System.out.println(filePath);
        return filePath;
    }

导出方法

自定义导出注解

定义注解后可根据中文注释自动给对应字段赋值

import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
 
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
 
/**
 * 实体model属性字段的位置顺序 ,从0开始
 */
@Documented
@Retention(RUNTIME)
@Target(FIELD)
public @interface BeanFieldIndex {
	 // 在表格中的列数
     int index();
	 // 列名
     String title();
}

导出的列表对象

import lombok.Data;

/**
 * 功能描述:
 *
 * @Author: liuxd
 * @description: 专家
 * @Date: 2023/2/2
 */
@Data
public class ExpertVo {

    /**
     * 序号
     */
    @BeanFieldIndex(index = 0,title = "序号")
    private Long id;
    
    /**
     * 姓名
     */
    @BeanFieldIndex(index = 1,title = "姓名")
    private String name;

    /**
     * 性别
     */
    @BeanFieldIndex(index = 2,title = "性别")
    private String sex;

    /**
     * 年龄
     */
    @BeanFieldIndex(index = 3,title = "年龄")
    private Integer age;

    /**
     * 科室
     */
    @BeanFieldIndex(index = 4,title = "科室")
    private String dept;

    /**
     * 所属部队
     */
    @BeanFieldIndex(index = 5,title = "所属部队")
    private String unit;

    /**
     * 地址
     */
    @BeanFieldIndex(index = 6,title = "地址")
    private String address;

import lombok.Data;

/**
 * 功能描述:
 *
 * @Author: liuxd
 * @description: 实体类字段与排序
 * @Date: 2023/2/2
 */
@Data
public class BeanVo {

    private String property;

    private String title;

    private int order;

    public BeanVo(){}

    public BeanVo(int index, String value,String property) {
        this.order = index;
        this.property = property;
        this.title = value;
    }
}
import com.deepoove.poi.data.TableRenderData;
import lombok.Data;


@Data
public class TableData {

    /**
     * 标题
     */
    private String title;

    /**
     * 表格
     */
    private TableRenderData table;

    private String[][] tableList;

    /**
     * 导出日期
     */
    private String exportDate;

}

/**
         * 功能描述:
         * 导出word,带表格
         * @Date: 2023-02-02 11:56:49
         * @Param tableData:
         * @Param list: 要导出的数据
         * @Param properties: 要导出的列
         * @Param clazz: 导出的对象
         * @Param fileName: 导出的文件名
         * @Param title: 标题栏内容
         * @Param sheetName: 导出的sheet名
         * @Param rule: 合并的单元格的规则
         * @Param headers: 表头
         * @Param response:
         * @return: void
         * @since: 1.0.0
         */
        public static void exportWord(List<?> list, String[] properties, String fileName, String title, String
                sheetName, MergeCellRule rule, String templateFile, HttpServletResponse response) throws Exception {

            TableData tableData = new TableData();
            tableData.setTitle(title);
            Object obj = list.get(0);
            Class<?> clazz = obj.getClass();
            Field[] fields = clazz.getDeclaredFields();
            // 表格标题栏
            String[] titleRow = new String[properties.length];
            // 标题栏的属性值 确保导出的顺序
            String[] titlePropertiesRow = new String[properties.length];
            List<BeanVo> beanVos = new ArrayList<>();
            for (int i = 0; i < properties.length; i++) {
                String property = properties[i];
                for (Field field : fields) {
                    // 找到当前匹配的字段
                    if (field.getName().equals(property)) {
                        BeanFieldIndex radiusIndex = field.getAnnotation(BeanFieldIndex.class);
                        if (radiusIndex != null) {
                            int index = radiusIndex.index();
                            String value = radiusIndex.title();
                            BeanVo beanVo = new BeanVo(index, value, property);
                            beanVos.add(beanVo);
                        }
                        break;
                    }
                }
            }
            // 根据order排序
            beanVos = beanVos.stream().sorted(Comparator.comparing(BeanVo::getOrder)).collect(Collectors.toList());
            for (int i = 0; i < properties.length; i++) {
                titleRow[i] = beanVos.get(i).getTitle();
                titlePropertiesRow[i] = beanVos.get(i).getProperty();
            }
            RowRenderData row10 = Rows.of("序号", "基本信息", "", "", "基本信息1", "", "").center().create();
            RowRenderData row0 = Rows.of(titleRow).center().create();
            // 表格数据 加上1行表头
            RowRenderData[] rowRenderData = new RowRenderData[list.size() + 2];
            rowRenderData[0] = row10;
            rowRenderData[1] = row0;
            // 单元格合并规则
//            MergeCellRule rule = mergeCell();
            // 表格内容赋值
            String[][] strings = new String[list.size()][properties.length];
            for (int i = 0; i < list.size(); i++) {
                Object o = list.get(i);
                strings[i] = new String[titlePropertiesRow.length];
                // 获取需要导出的字段值
                for (int j = 0; j < titlePropertiesRow.length; j++) {
                    // 拼接该字段对应的get方法的方法名
                    String getMethodName = "get" + titlePropertiesRow[j].substring(0, 1).toUpperCase() + titlePropertiesRow[j].substring(1);
                    Method getMethod = clazz.getMethod(getMethodName, new Class[]{});
                    try {
                        // 利用Java反射机制调用get方法获取字段对应的值
                        Object target = getMethod.invoke(o, new Object[]{});
                        strings[i][j] = String.valueOf(target);
                    } catch (IllegalAccessException | InvocationTargetException | IllegalArgumentException e) {
                        System.out.println(e);
                    }
                }
                rowRenderData[i + 2] = Rows.of(strings[i]).center().create();
            }
            tableData.setTableList(strings);

            TableRenderData table = Tables.of(rowRenderData).mergeRule(rule).create();
            // 数据封装
            tableData.setTable(table);
            // 传入模板模板地址+信息数据
            XWPFTemplate template = XWPFTemplate.compile(templateFile).render(tableData);
            // 指定下载的文件名--设置响应头
            response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));
            response.setContentType("application/vnd.ms-word;charset=UTF-8");
            response.setHeader("Pragma", "no-cache");
            response.setHeader("Cache-Control", "no-cache");
            response.setDateHeader("Expires", 0);
            OutputStream out = response.getOutputStream();
            BufferedOutputStream bos = new BufferedOutputStream(out);
            try {
                template.write(out);
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                bos.flush();
                out.flush();
                bos.close();
                out.close();
                template.close();
            }
        }

单元格合并规则

/***
         * 功能描述:
         * 导出word中的表格时合并相应单元格
         * 要根据具体的需求去根据不同字段 分组 来确定单元格合并的行数与列数
         * @Date: 2023-02-02 18:11:11
         * @Param
         * @return: com.deepoove.poi.data.MergeCellRule
         * @since: 1.0.0
         */
    public static MergeCellRule mergeCell() {
        // 合并后单元格
        /*MergeCellRule rule = MergeCellRule.builder().map(MergeCellRule.Grid.of(0, 0), MergeCellRule.Grid.of(1, 0)).
                // 从第几行第几列合并到第几行第几列
                map(MergeCellRule.Grid.of(0, 1), MergeCellRule.Grid.of(0, 3)).
                map(MergeCellRule.Grid.of(0, 4), MergeCellRule.Grid.of(0, 6)).
//                    map(MergeCellRule.Grid.of(0, 4), MergeCellRule.Grid.of(1, 4)).
//                    map(MergeCellRule.Grid.of(rowRenderData.length - 1, 0), MergeCellRule.Grid.of(rowRenderData.length - 1, 3)).
                build();*/
        MergeCellRule.MergeCellRuleBuilder rule = MergeCellRule.builder();
        rule.map(MergeCellRule.Grid.of(0, 1), MergeCellRule.Grid.of(0, 3));
        rule.map(MergeCellRule.Grid.of(0, 4), MergeCellRule.Grid.of(0, 6));
        return rule.build();
    }

测试

/***
     * 功能描述:
     * 根据用户ID更新ws推送的参数
     * @Date: 2022-12-01 09:21:25
     * @return: void
     * @since: 1.0.0
     */
    @RequestMapping("/download")
    public void download(HttpServletResponse response){
        try {
            List<ExpertVo> expertVoList = new ArrayList<>();
            expertVoList.add(new ExpertVo(Long.valueOf("1"),"张三1","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("2"),"张三2","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("3"),"张三3","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("4"),"张三4","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("5"),"张三5","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("6"),"张三6","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("7"),"张三7","男",28,"","职务1","地址1"));
            expertVoList.add(new ExpertVo(Long.valueOf("8"),"张三8","男",28,"","职务1","地址1"));
            String templateFile = getTemplateAbsolutePath("wordtemplate/wordexporttemplate.docx");
            // new String[]{"id","name","sex","dept","unit","address","age"} 为需要导出的列
            WordExcelUtil.exportWord(expertVoList,new String[]{"id","name","sex","dept","unit","address","age"},"专家信息导出.docx","专家信息",null,WordExcelUtil.mergeCell(),templateFile,response);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (EncodeException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

导入方法

导入带合并单元格的思路为 当前单元格无值,就向上找,直到找到有值的单元格,将该单元格的值赋给当前单元格
.doc与.docx导入的方法不同

/**
     * 功能描述:
     * 导入带表格的word文档
     * @Date: 2023-02-03 16:55:56
     * @Param  file:
     * @Param firstRow: 从第几行开始读(具体数据行的上一行)
     * @Param clazz: 表格对象的实体类
     * @Param: LinkedHashMap<String, Object> map   Word中除excel表格外的其他段落的内容 传参new 一个Map
     * @return: java.util.List<T>
     * @since: 1.0.0
     */
    public static <T> List<T> importWord(MultipartFile file, int firstRow, Class clazz, LinkedHashMap<String, Object> map,String fileName) throws Exception {
        if(fileName.endsWith(".doc")){
            return doDoc(file,firstRow,clazz,map);
        }else if(fileName.endsWith(".docx")){
            return doDocx(file,firstRow,clazz,map);
        }else{
            return null;
        }
    }

doc导入

/***
     * 功能描述:
     * 读取doc文档中表格数据示例 这里是03版 doc格式表格的读取方法
     * @Author: LXD
     * @Date: 2023-02-03 16:57:19
     * @Param  file:
     * @Param firstRow: 从第几行开始读(具体数据行的上一行)
     * @Param clazz: 表格对象的实体类
     * @Param: LinkedHashMap<String, Object> map   Word中除excel表格外的其他段落的内容 传参new 一个Map
     * @return: java.util.List<T>
     * @since: 1.0.0
     */
    public static <T> List<T> doDoc(MultipartFile file, int firstRow, Class clazz, LinkedHashMap<String, Object> map) throws Exception {
        //用于存放预览信息
        List<String> stringList = new ArrayList<String>();
        List<T> list = new ArrayList<T>();
        try {
            //获取文件流
            InputStream is = file.getInputStream();
            //获取文件对象
            HWPFDocument doc = new HWPFDocument(is);
            //获取文件内容对象
            Range r = doc.getRange();
            //获取文件中所有的表格
            TableIterator it = new TableIterator(r);
            for (int i = 0; i < r.numParagraphs(); i++) {
                //获取当前段落
                Paragraph p = r.getParagraph(i);
                //判断当前段落是否为表格
                if (p.isInTable()) {
                    int rowNum = 0 + firstRow==1? 0 : firstRow-1;
                    // 先读取表头信息
                    List<String> firstRowStrings = new ArrayList<>();
                    while (it.hasNext()) {
                        Table tb = (Table) it.next();
                        // 循环所有行
                        while(rowNum<firstRow) {
                            //第一行
                            TableRow firstTr = tb.getRow(firstRow-1);
                            //迭代文档中的表格
                            for (int x = 0; x < firstTr.numCells(); x++) {
                                //取得单元格
                                TableCell td = firstTr.getCell(x);
                                //取得单元格的内容
                                Paragraph para = td.getParagraph(0);
                                String text = para.text();
                                if (StringUtils.isEmpty(text) || "\u0007".equals(text)) {
                                    text = tb.getRow(0).getCell(x).getParagraph(0).text();
                                }
                                if(StringUtils.isNotBlank(text)){
                                    // 去除不可见字符
                                    text = text.replaceAll("\u0007","");
                                }
                                firstRowStrings.add(text);
                            }
                            rowNum++;
                        }
                        while(rowNum>=firstRow){
                            //迭代行,默认从firstRow开始 // 跳过表头信息行
                            for (int j = firstRow; j < tb.numRows(); j++) {
                                List<String> rowStrings = new ArrayList<>();
                                //当前行
                                TableRow tr = tb.getRow(j);
                                //用于存放一行数据,不需要可以不用
                                String rowText = "";
                                T t = (T) clazz.newInstance();
                                //迭代列,默认从0开始
                                for (int x = 0; x < tr.numCells(); x++) {
                                    //取得单元格
                                    TableCell td = tr.getCell(x);
                                    //取得单元格的内容
                                    Paragraph para = td.getParagraph(0);
                                    String text = para.text();
                                    int times = 1;
                                    while (StringUtils.isEmpty(text) || "\u0007".equals(text)){
                                        TableRow preTr = tb.getRow(j-times);
                                        TableCell preCell = preTr.getCell(x);
                                        //取得单元格的内容
                                        text = preCell.getParagraph(0).text();
                                        times = times + 1;
                                    }
                                    if(StringUtils.isNotBlank(text)){
                                        // 去除不可见字符
                                        text = text.replaceAll("\u0007","");
                                    }
                                    //自己用“ ”区分两列数据,根据自己需求 可以省略
                                    if (StringUtils.isEmpty(rowText)) {
                                        rowText = text;
                                        rowStrings.add(text);
                                    } else {
                                        rowText = rowText + " " + text;
                                        rowStrings.add(text);
                                    }
                                }
                                stringList.add(rowText);
                                BeanRefUtils.setProperty(t, rowStrings,firstRowStrings);
                                list.add(t);
                            }
                            break;
                        }
                    }
                }else{
                    System.out.println(p.text());
                    if(StringUtils.isNotBlank(p.text())){
                        map.put(String.valueOf(i+1),p.text());
                    }
                }
            }
            return list;
        } catch (IOException e) {
            log.error("[文件操作 - 读取03版doc文件] - 文件读取失败,文件名称:{}", file.getOriginalFilename());
            throw new Exception("文件读取失败", e);
        }
    }

docx导入

/***
         * 功能描述:
         * 读取doc文档中表格数据示例 这里是07版 docx格式表格的读取方法
         * @Date: 2023-02-02 19:11:23
         * @Param  file:
         * @Param firstRow: 从第几行开始读(具体数据行的上一行)
         * @Param: LinkedHashMap<String, Object> map   Word中除excel表格外的其他段落的内容 传参new 一个Map
         * @since: 1.0.0
         */
        public static <T> List<T> doDocx (MultipartFile file,int firstRow,Class clazz,LinkedHashMap<String, Object> map) throws Exception {
            //用于存放预览信息
            List<String> stringList = new ArrayList<String>();
            List<T> list = new ArrayList<T>();
            try {
                //获取文件流
                InputStream is = file.getInputStream();
                //获取文件对象
                XWPFDocument doc = new XWPFDocument(is);
                //获取文件内容对象
                XWPFParagraph[] paras = doc.getParagraphs().toArray(new XWPFParagraph[0]);
                // 获取所有段落内容
                Iterator<XWPFParagraph> itPara = doc.getParagraphsIterator();//获取word段落
                // 段落序号
                int paragraphNum = 1;
                while (itPara.hasNext()) {
                    XWPFParagraph paragraph = itPara.next();//获取段落
                    // 非表格
                    if(!paragraph.getPartType().equals(BodyType.TABLECELL)){
                        System.out.println(paragraph.getText());
                        if(StringUtils.isNotBlank(paragraph.getText())){
                            map.put(String.valueOf(paragraphNum),paragraph.getText());
                        }
                        paragraphNum++;
                //获取所有的表格对象
                List tables = doc.getTables();
                //因我文档中只有一个所以这里没有去遍历,直接获取指定表格对象
                XWPFTable table = (XWPFTable) tables.get(0);
                // 获取表头信息
                List<String> firstRowStrings = new ArrayList<>();
                //表头行信息
                XWPFTableRow firstTr = table.getRow(1);
                for (int x = 0; x < firstTr.getTableCells().size(); x++) {
                    //取得单元格
                    XWPFTableCell cell = firstTr.getCell(x);
                    //取得单元格的内容
                    String text = cell.getText();
                    // 如果表头该列为空,则读取该列上一行的数据
                    if(StringUtils.isEmpty(text)){
                        text = table.getRow(0).getCell(x).getText();
                    }
                    firstRowStrings.add(text);
                }
                //迭代行,默认从0开始 (第一行若为表头,则可以跳过)
                for (int j = firstRow; j < table.getRows().size(); j++) {
                    List<String> rowStrings = new ArrayList<>();
                    //当前行
                    XWPFTableRow tr = table.getRow(j);
                    //用于存放一行数据,不需要可以不用
                    String rowText = "";
                    T t = (T) clazz.newInstance();
                    //迭代列,默认从0开始
                    for (int x = 0; x < tr.getTableCells().size(); x++) {
                        //取得单元格
                        XWPFTableCell cell = tr.getCell(x);
                        //取得单元格的内容
                        String text = cell.getText();
                        //自己用“ ”区分两列数据,根据自己需求 可以省略
                        int times = 1;
                        while (StringUtils.isEmpty(text)){
                            XWPFTableRow preTr = table.getRow(j-times);
                            XWPFTableCell preCell = preTr.getCell(x);
                            //取得单元格的内容
                            text = preCell.getText();
                            times = times + 1;
                        }
                        if (StringUtils.isEmpty(rowText)) {
                            rowText = text;
                            rowStrings.add(text);
                        } else {
                            rowStrings.add(text);
                            rowText = rowText + " ;" + text;
                        }
                    }
                    stringList.add(rowText);
                    BeanRefUtils.setProperty(t, rowStrings,firstRowStrings);
                    list.add(t);
                }
                return list;
            } catch (IOException e) {
                log.warn("[文件操作 - 读取07版docx文件] - 文件读取失败,文件名称:{}", file.getOriginalFilename());
                throw new Exception("文件读取失败", e);
            }
        }

测试导入

    public static void main(String[] args) throws Exception {
        System.err.println("开始读取");
        String filePath = "C:\\Users\\XXX\\Desktop\\2023专家信息.doc";
        File file = new File(filePath);
        try {
            MultipartFile cMultiFile = new MockMultipartFile("file", file.getName(), null, new FileInputStream(file));
            // 除表格外的段落内容
            LinkedHashMap<String, Object> map = new LinkedHashMap<>();
			importWord(cMultiFile,2,ExpertVo.class,map,file.getName()).stream().forEach(System.out::println);
            System.out.println(map.toString());
            System.err.println("结束读取");
        } catch (IOException e) {
            e.printStackTrace();
        }
      }
;