最近项目快了验收,那么接下来就是写一些比较烦人的文档,在写数据库设计文档时,到了详细设计这一块有点尴尬了,每张表,没个字段都要写上去,由于前期没有整理,所以这个工作量还是很大,我查看了我们的数据库发现有353张表,这样写,得花多久的时间啊。。。于是想通过程序来自动完成,这就是这篇文章的核心。
系列文章: 自动生成数据库表设计(一)之获取JDBC获取元数据 自动生成数据库表设计(二)之Freemarker的基本使用 自动生成数据库表设计(三)之制作word模版
本篇主要内容:
1、配置Maven工程
2、数据库元数据的获取
配置Maven工程
1、新建Maven工程
1.创建工程我这里创建的moudle类似于Eclipse的工程
2.指定Moudle路径
3.点击OK,编辑pom.xml (无所谓)
4.在main下新建lib文件夹,我们要拷贝Oracle的jdbc的jar包,我没找到maven依赖
2、配置工程
1.创建好工程后,开始配置工程,由于我们的数据库是Oracle11g,这里我们先找到Oracle的JDBC的jar包,找到 安装目录下的product\11.2.0\dbhome_1\jdbc\lib 的odbc6.jar拷贝到工程的lib下
2.将jar包添加到依赖
选中工程-右键-open moudle settings
第4步选中我们odbc6所在的目录,也就是刚才新建的lib包
至此我们的工程配置完成。
数据库元数据的获取
1、预览模版样式确定需求(前提是你的数据库已经存在这些信息)
获取的内容有:
表:表名、表注释
列:列名、列注释、字段类型、默认值
那么我们就要获取数据库的这些东西。
根据表名获取列信息的SQL
SELECT
utc.table_name,utc.column_name,utc.data_type,utc.data_length,utc.data_default,ucc.comments,p.PRIMARY_KEY
FROM
user_tab_columns utc
LEFT JOIN --查询注释
user_col_comments ucc
ON
utc.table_name = ucc.table_name
AND
utc.column_name = ucc.column_name
LEFT JOIN --查询主键
(
SELECT
col.table_name table_name,
col.column_name column_name,
CASE con.constraint_type WHEN 'P' THEN 'true' ELSE 'false' END "PRIMARY_KEY"
FROM
user_constraints con,
user_cons_columns col
WHERE
con.constraint_name = col.constraint_name
AND
con.constraint_type = 'P'
) p
ON
utc.column_name = p.column_name
AND
p.table_name = utc.table_name
WHERE
utc.table_name = 'YJ_HL';
结果如下:
获取数据库表信息的SQL
SELECT * FROM user_tab_comments WHERE table_type='TABLE'
这里就不显示了,设计公司业务了
ok现在我们基本信息用sql已经完成,那么用jdbc执行这些sql基本就完成获取数据库元数据信息了。
3、编写JDBC获取元数据信息
getTableList如下
// 获取数据库中所有表的表名,并添加到列表结构中。
public static List getTableList(Connection conn) throws SQLException {
List<Map> tableList = new ArrayList<Map>();
String sql =
"SELECT * FROM user_tab_comments WHERE table_type='TABLE'";
PreparedStatement ps = conn.prepareStatement(sql);
ResultSet rs = ps.executeQuery();
while (rs.next()) {
Map map = new HashMap<>();
String TABLE_NAME = rs.getString("TABLE_NAME");
String COMMENTS = rs.getString("COMMENTS");
map.put("TABLE_NAME", TABLE_NAME);
map.put("COMMENTS", COMMENTS == null ? "" : COMMENTS);
//获取列
List<Map> columnList = getColumnList(conn, TABLE_NAME);
map.put("COLUMNS", columnList);
//这里是过滤特殊的表,比如只生成SYS开头的表
if (TABLE_NAME.startsWith("SYS"))
tableList.add(map);
System.out.println("TABLE_NAME ==>" + TABLE_NAME + " COMMENTS==>" + COMMENTS);
}
rs.close();
ps.close();
return tableList;
}
getColumnList如下
// 获取数据表中所有列的列名,并添加到列表结构中。
public static List getColumnList(Connection conn, String tableName)
throws SQLException {
List<Map> columnList = new ArrayList<Map>();
String sql =
"SELECT utc.table_name,utc.column_name,utc.data_type,utc.data_length,utc.data_default,utc.nullable,ucc.comments,p.PRIMARY_KEY " +
"FROM user_tab_columns utc " +
"LEFT JOIN user_col_comments ucc " + //--查询注释
"ON utc.table_name = ucc.table_name " +
"AND utc.column_name = ucc.column_name " +
"LEFT JOIN " + //--查询主键
"( SELECT col.table_name table_name, col.column_name column_name, CASE con.constraint_type WHEN 'P' THEN 'true' ELSE 'false' END PRIMARY_KEY " +
"FROM user_constraints con,user_cons_columns col " +
"WHERE con.constraint_name = col.constraint_name " +
"AND con.constraint_type = 'P') p " +
"ON utc.column_name = p.column_name " +
"AND p.table_name = utc.table_name " +
"WHERE utc.table_name = ?";
PreparedStatement ps = conn.prepareStatement(sql);
ps.setString(1, tableName);
ResultSet rs = ps.executeQuery();
while (rs.next()) {
Map map = new HashMap<>();
String COLUMN_NAME = rs.getString("COLUMN_NAME");
String DATA_TYPE = rs.getString("DATA_TYPE");//VARCHAR2
String DATA_LENGTH = rs.getString("DATA_LENGTH");//200
String DATA_DEFAULT = rs.getString("DATA_DEFAULT");
String NULLABLE = rs.getString("NULLABLE");
String COMMENTS = rs.getString("COMMENTS");
String PRIMARY_KEY = rs.getString("PRIMARY_KEY");
map.put("COLUMN_NAME", COLUMN_NAME);
map.put("DATA_TYPE", DATA_TYPE);
map.put("DATA_LENGTH", DATA_LENGTH);
map.put("DATA_DEFAULT", DATA_DEFAULT == null ? "" : DATA_DEFAULT);
map.put("NULLABLE", "N".equals(NULLABLE) ? false : true);
map.put("COMMENTS", COMMENTS == null ? "" : COMMENTS);
map.put("PRIMARY_KEY", "true".equals(PRIMARY_KEY) ? true : false);
columnList.add(map);
System.out.println("COLUMN_NAME ==>" + COLUMN_NAME + " DATA_TYPE==>" + DATA_TYPE + " DATA_LENGTH==>" + DATA_LENGTH + " NULLABLE==>" + NULLABLE + " COMMENTS==>" + COMMENTS + " PRIMARY_KEY==>" + PRIMARY_KEY);
}
rs.close();
ps.close();
return columnList;
}
4、测试
public static void main(String[] args) throws Exception {
Class.forName("oracle.jdbc.driver.OracleDriver");
Connection conn = DriverManager.getConnection("jdbc:oracle:thin:@127.0.0.1:1521:orcl", "scott", "tiger");
List<Map> tableList = getTableList(conn);
conn.close();
FtUtil ftUtil = new FtUtil();
Map map = new HashMap<>();
map.put("table", tableList);
ftUtil.generateFile("/", "moban.xml", map, "D:/", "sys_moban.doc");
}
打印结果如下:
至此,我们基本完成需求中所需要的数据了
下一篇,我们讲Freemarker简单的使用。