文章目录
- 1.场景说明
- 2.解决办法
- 3.生成文档
- 3.1.实现思路
- 3.2.引入Apache POI依赖
- 3.3.获取表及表字段说明Mapper
- 3.4.POI创建文档表格,并填充数据
- 3.5.完整的接口下载代码
- 3.6.效果展示
1.场景说明
在项目中表已经建立好了,但是现在想对外提供一个表的字段的描述说明,该怎么办。例如开发前期赶进度,也没个什么数据库的需求,设计文档之类的,项目开发后期,现在要补充文档了,表要是少,那就挨个复制粘贴了,多的话,复制粘贴也麻烦。
例如,这是某个表
希望最后在word文档中提供这样的表结构说明
2.解决办法
1.根据建表的语句挨个的复制呗,还能怎么办,适用于表量比较少,然后技术比较菜的人,如果数据库的表有个七八十,上百张,相信我,真的有人挨个的将SQL客户端工具打开,然后挨个的复制到文档里面去。然后时候说自己工作很多,加班很晚才弄完。
2.有点学习技术需求,不至于太菜的,虽然我没去找过有没有,但是我相信这种网上肯定有一堆人已经实现将数据库表结构转换成word文档的代码或者示例,所以我觉得,只要不是太菜,一下午也能解决了。
3.直接让GPT转换了,导出建表语句,给GPT一个示例,然后GPT来做
例如如下,导出一个建表语句,然后再最上面给出一个示例,然后接下来就交给GPT了
4.接下来才是本文的重头戏,带着学习的成分,我们去造轮子,当然了,不是造GPT,咱没有那个本事,咱们自己写一个组件来实现,根据数据库逆向生成表结构的文档说明
造轮子,根据数据库表生成文档
3.生成文档
3.1.实现思路
1.获取到所有的表,根据你使用的数据库,查询表,例如我用mysql,因此用下面的语句查看表名
SHOW TABLES;
2.获取表的字段描述说明
SHOW FULL FIELDS FROM <表名>
3.利用步骤1和步骤2的表名、表结构信息,生成多个word表格,
java中生成word的技术有很多
-
Apache POI:Apache POI是一个开源的Java库,用于读取和写入Microsoft
Office格式的文件,包括Word文档(.doc和.docx)。它提供了丰富的API和类,可以操作和处理Word文档的内容、格式、样式和元数据等。 -
Docx4j:Docx4j是一个用于创建和操作.docx文件的Java库。它提供了许多API和类,可以通过编程方式创建、修改和处理Word文档。Docx4j还支持生成PDF、HTML和其他格式的文档。
-
Apache POI XWPF:Apache POI XWPF是Apache
POI库的扩展,专门用于处理.docx格式的Word文档。它提供了更高级的API,可以读取、写入和修改.docx文件,并支持处理文本、样式、段落、表格、图像和其他文档元素。 -
JWord:JWord是一个商业库,用于创建和操作Word文档。它提供了丰富的API和功能,可以生成复杂的Word文档,包括文本、样式、表格、图像、图表和其他元素。
-
Aspose.Words for Java:Aspose.Words是一个商业库,用于在Java应用程序中处理Word文档。它提供了强大的API和功能,可以创建、修改、转换和打印Word文档,并支持许多高级特性,如合并文档、插入水印、执行邮件合并等
本文中,我们采用Apache POI来实现
3.2.引入Apache POI依赖
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>4.1.2</version>
</dependency>
3.3.获取表及表字段说明Mapper
定义实体类TableStruct接收我们需要的字段,例如,这里我们需要field、type、comment上字段
@Data
public class TableStruct {
private String field;
private String type;
private String comment;
}
编写我们的SQL,这里我使用Mybatis,当然,其他的SQL框架也行
public interface TableMapper {
@Select("SHOW TABLES")
List<String> getAllTables();
@Select("SHOW FULL FIELDS FROM ${tableName}")
List<TableStruct> getTableInfo(@Param("tableName") String tableName);
}
3.4.POI创建文档表格,并填充数据
//设置标题
private static void addCustomHeadingStyle(XWPFStyles styles, String styleId, String styleName, int headingLevel) {
CTStyle ctStyle = CTStyle.Factory.newInstance();
ctStyle.setStyleId(styleId);
CTString styleNameString = CTString.Factory.newInstance();
styleNameString.setVal(styleName);
ctStyle.setName(styleNameString);
CTDecimalNumber indentNumber = CTDecimalNumber.Factory.newInstance();
indentNumber.setVal(BigInteger.valueOf(headingLevel));
CTPPr ppr = CTPPr.Factory.newInstance();
ppr.setOutlineLvl(indentNumber);
ctStyle.setPPr(ppr);
XWPFStyle style = new XWPFStyle(ctStyle);
styles.addStyle(style);
}
// 创建文档
XWPFDocument document = new XWPFDocument();
//设置标题
XWPFStyles styles = document.createStyles();
String heading1StyleId = "heading1";
addCustomHeadingStyle(styles, heading1StyleId, "标题 1", 1);
tableMapper.getAllTables().forEach(temp ->{
//生成标题
XWPFParagraph title1Paragraph = document.createParagraph();
title1Paragraph.setStyle(heading1StyleId);
XWPFRun title1Run = title1Paragraph.createRun();
title1Run.setText(temp);
//生成表头
List<TableStruct> tableInfo = tableMapper.getTableInfo(temp);
XWPFTable table = document.createTable(tableInfo.size() + 1, 4);
table.setWidth("100%");
table.getRow(0).getCell(0).setText("序号");
table.getRow(0).getCell(1).setText("字段名称");
table.getRow(0).getCell(2).setText("字段类型");
table.getRow(0).getCell(3).setText("字段描述");
//生成表内容 第0行已经设置为表头,因此我们需要从第一行开始
for (int row = 0; row < tableInfo.size(); row++) {
table.getRow(row +1 ).getCell(0).setText(Integer.toString(row));
table.getRow(row +1 ).getCell(1).setText(tableInfo.get(row).getField());
table.getRow(row +1 ).getCell(2).setText(tableInfo.get(row).getType());
table.getRow(row +1 ).getCell(3).setText(tableInfo.get(row).getComment());
}
});
3.5.完整的接口下载代码
@GetMapping("/info/download")
public void downSeg(HttpServletResponse response){
try {
response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
response.setCharacterEncoding("utf-8");
String fileName = URLEncoder.encode("表结构注释说明"+LocalDate.now().toString(), "UTF-8").replaceAll("\\+", "%20");
response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName + ".docx");
// 创建文档
XWPFDocument document = new XWPFDocument();
//设置标题
XWPFStyles styles = document.createStyles();
String heading1StyleId = "heading1";
addCustomHeadingStyle(styles, heading1StyleId, "标题 1", 1);
tableMapper.getAllTables().forEach(temp ->{
//生成标题
XWPFParagraph title1Paragraph = document.createParagraph();
title1Paragraph.setStyle(heading1StyleId);
XWPFRun title1Run = title1Paragraph.createRun();
title1Run.setText(temp);
//生成表头
List<TableStruct> tableInfo = tableMapper.getTableInfo(temp);
XWPFTable table = document.createTable(tableInfo.size() + 1, 4);
table.setWidth("100%");
table.getRow(0).getCell(0).setText("序号");
table.getRow(0).getCell(1).setText("字段名称");
table.getRow(0).getCell(2).setText("字段类型");
table.getRow(0).getCell(3).setText("字段描述");
//生成表内容 第0行已经设置为表头,因此我们需要从第一行开始
for (int row = 0; row < tableInfo.size(); row++) {
table.getRow(row +1 ).getCell(0).setText(Integer.toString(row));
table.getRow(row +1 ).getCell(1).setText(tableInfo.get(row).getField());
table.getRow(row +1 ).getCell(2).setText(tableInfo.get(row).getType());
table.getRow(row +1 ).getCell(3).setText(tableInfo.get(row).getComment());
}
});
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
document.write(byteArrayOutputStream);
ServletOutputStream outputStream = response.getOutputStream();
outputStream.write(byteArrayOutputStream.toByteArray());
outputStream.flush();
outputStream.close();
document.close();
} catch (Exception e) {
e.printStackTrace();
}
}
private static void addCustomHeadingStyle(XWPFStyles styles, String styleId, String styleName, int headingLevel) {
CTStyle ctStyle = CTStyle.Factory.newInstance();
ctStyle.setStyleId(styleId);
CTString styleNameString = CTString.Factory.newInstance();
styleNameString.setVal(styleName);
ctStyle.setName(styleNameString);
CTDecimalNumber indentNumber = CTDecimalNumber.Factory.newInstance();
indentNumber.setVal(BigInteger.valueOf(headingLevel));
CTPPr ppr = CTPPr.Factory.newInstance();
ppr.setOutlineLvl(indentNumber);
ctStyle.setPPr(ppr);
XWPFStyle style = new XWPFStyle(ctStyle);
styles.addStyle(style);
}
3.6.效果展示
浏览器访问该接口,即可下载
效果如下所示
OK,收拾东西,准备下班(2023年11月28日16:57:24),楼主单位4点50下班,从产生这个想法,到我这个想法实现,以及到我生成这个文档以后,将这50多张表的结构说明,补充到详细设计文档里面,总共花了大概2个小时。今天下午又是收货满满的一天。