引言
本篇将要演示 FleaJDBCHelper 的使用,该工具类封装了基本的JDBC增删改查的操作,只需简单几步即可实现数据库操作。
1. 准备工作
为了演示JDBC接入(参考 JPA接入 中的准备工作),需要如下准备:
- MySQL数据库 (客户端可以使用 navicat for mysql)
- 新建测试数据库 fleajpatest
- 新建测试表 student
2. 接入讲解
2.1 JDBC数据源配置
数据源配置独立出来,定义在 flea-config.xml 中,可添加多个JDBC数据库配置【即 config-items 节点】。
1 2 3 4 5 6 7 8 9 10 11 12 13
| <?xml version="1.0" encoding="UTF-8"?>
<flea-config>
<config-items key="mysql-fleajpatest" desc="JDBC数据库配置【key=数据库系统-数据库或数据库用户】"> <config-item key="driver" desc="mysql数据库驱动名">com.mysql.jdbc.Driver</config-item> <config-item key="url" desc="mysql数据库连接地址">jdbc:mysql://localhost:3306/fleajpatest?useUnicode=true&characterEncoding=UTF-8</config-item> <config-item key="user" desc="mysql数据库登录用户名">root</config-item> <config-item key="password" desc="mysql数据库登录密码">root</config-item> </config-items>
</flea-config>
|
2.2 定义Flea数据库单元
每个 FleaDBUnit 中, database 和 name 对应上述 config-items 中的 key;driver、url、user、password 分别对应上述 config-item 中的配置。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
public class FleaDBUnit {
private String database; private String name; private String driver; private String url; private String user; private String password;
}
|
database
: 数据库管理系统名
name
: 数据库名 或 数据库用户名
driver
: 数据库驱动名
url
: 数据库连接地址
user
: 数据库登录用户名
password
: 数据库登录密码
2.3 定义Flea数据库操作类
FleaDBOperation 封装了JDBC的数据库操作对象,包括数据库连接对象 Connection、数据库预编译状态对象 PreparedStatement 和 数据库结果集对象 ResultSet。该类继承 Closeable,实现 close 方法,用于每次 JDBC 数据库操作后释放资源【这里用到了 try-with-resource 语法糖 】。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
|
public class FleaDBOperation implements Closeable {
private Connection connection;
private PreparedStatement preparedStatement;
private ResultSet resultSet;
@Override public void close() { FleaJDBCConfig.close(connection, preparedStatement, resultSet); } }
|
2.4 定义Flea JDBC配置类
FleaJDBCConfig 读取数据库的配置信息,该信息存在于 flea-config.xml
中。 其中的 init(String mDatabase, String mName) 方法用于初始化本次操作的数据库管理系统【mDatabase】、数据库名或数据库用户 【mName】,两者对应数据库配置 config-items 中的 key 。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
| public class FleaJDBCConfig {
private static volatile FleaJDBCConfig config;
private static final ConcurrentMap<String, FleaDBUnit> fleaDBUnits = new ConcurrentHashMap<>();
private static final Object fleaDBUnitsLock = new Object();
private FleaJDBCConfig() { }
public static FleaJDBCConfig getConfig() {
if (ObjectUtils.isEmpty(config)) { synchronized (FleaJDBCConfig.class) { if (ObjectUtils.isEmpty(config)) { config = new FleaJDBCConfig(); } } } return config; }
public static void init(String mDatabase, String mName) { FleaFrameManager.getManager().setDBConfigKey(mDatabase, mName); }
public Connection getConnection() { Connection conn = null; FleaDBUnit fleaDBUnit;
String dbConfigKey = FleaFrameManager.getManager().getDBConfigKey();
if (!fleaDBUnits.containsKey(dbConfigKey)) { synchronized (fleaDBUnitsLock) { if (!fleaDBUnits.containsKey(dbConfigKey)) { fleaDBUnits.put(dbConfigKey, getFleaDBUnit(dbConfigKey)); } } }
fleaDBUnit = fleaDBUnits.get(dbConfigKey);
try { ObjectUtils.checkEmpty(fleaDBUnit, DaoException.class, "ERROR-DB-DAO0000000013"); Class.forName(fleaDBUnit.getDriver()); conn = DriverManager.getConnection(fleaDBUnit.getUrl(), fleaDBUnit.getUser(), fleaDBUnit.getPassword()); } catch (Exception e) { LOGGER.error("获取数据库连接异常 :{}", e.getMessage()); } return conn; }
private FleaDBUnit getFleaDBUnit(String dbConfigKey) { FleaDBUnit fDBUnit = null; if (StringUtils.isNotBlank(dbConfigKey)) { fDBUnit = new FleaDBUnit(); String[] dbConfigKeyArr = StringUtils.split(dbConfigKey, CommonConstants.SymbolConstants.HYPHEN); if (ArrayUtils.isNotEmpty(dbConfigKeyArr) && CommonConstants.NumeralConstants.INT_TWO == dbConfigKeyArr.length) { fDBUnit.setDatabase(dbConfigKeyArr[0]); fDBUnit.setName(dbConfigKeyArr[1]); } fDBUnit.setDriver(FleaConfigManager.getConfigItemValue(dbConfigKey, DBConfigConstants.DB_CONFIG_DRIVER)); fDBUnit.setUrl(FleaConfigManager.getConfigItemValue(dbConfigKey, DBConfigConstants.DB_CONFIG_URL)); fDBUnit.setUser(FleaConfigManager.getConfigItemValue(dbConfigKey, DBConfigConstants.DB_CONFIG_USER)); fDBUnit.setPassword(FleaConfigManager.getConfigItemValue(dbConfigKey, DBConfigConstants.DB_CONFIG_PASSWORD)); }
return fDBUnit; }
private static void closeConnection(Connection conn) { try { if (ObjectUtils.isNotEmpty(conn)) { conn.close(); } } catch (SQLException e) { } }
private static void closeStatement(Statement statement) { try { if (ObjectUtils.isNotEmpty(statement)) { statement.close(); } } catch (SQLException e) { } }
private static void closeResultSet(ResultSet rs) { try { if (ObjectUtils.isNotEmpty(rs)) { rs.close(); } } catch (SQLException e) { } }
public static void close(Connection conn, Statement statement, ResultSet rs) { closeResultSet(rs); closeStatement(statement); closeConnection(conn); } }
|
上述 init 方法中,使用了 FleaFrameManager.getManager().setDBConfigKey,这里是将 JDBC连接的数据库配置添加到了线程对象中 ThreadLocal 中,代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| private static ThreadLocal<String> sDBLocal = new ThreadLocal<>();
public String getDBConfigKey() { return sDBLocal.get(); }
public void setDBConfigKey(String dbSysName, String dbName) { if (StringUtils.isNotBlank(dbSysName) && StringUtils.isNotBlank(dbName)) { sDBLocal.set(dbSysName.toLowerCase() + CommonConstants.SymbolConstants.HYPHEN + dbName.toLowerCase()); } }
|
2.4 定义 Flea JDBC工具类
在使用 FleaJDBCHelper 之前,一定要先调用一下 init。
3. 接入自测
请查看单元测试类 StudentJDBCTest
3.1 JDBC新增数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| @Test public void testStudentInsert() throws Exception { FleaJDBCConfig.init(DBSystemEnum.MySQL.getName(), "fleajpatest");
String sql = "insert into student(stu_name, stu_age, stu_sex, stu_state) values(?, ?, ?, ?)";
List<Object> paramList = new ArrayList<Object>(); paramList.add("huazie"); paramList.add(25); paramList.add(1); paramList.add(1);
int ret = FleaJDBCHelper.insert(sql, paramList);
LOGGER.debug("RESULT = {}", ret); }
|
执行结果:
3.2 JDBC查询数据
这里的查询语句,可以是复杂SQL,返回结果 List<Map<String, Object>>
1 2 3 4 5 6 7 8 9 10 11
| @Test public void testStudentQuery() throws Exception { FleaJDBCConfig.init(DBSystemEnum.MySQL.getName(), "fleajpatest");
String sql = "select * from student where stu_state = ?";
List<Object> paramList = new ArrayList<Object>(); paramList.add(1);
LOGGER.debug("RESULT LIST = {}", FleaJDBCHelper.query(sql, paramList)); }
|
执行结果:
1 2 3 4 5 6 7 8 9 10 11
| @Test public void testStudentSingleQuery() throws Exception { FleaJDBCConfig.init(DBSystemEnum.MySQL.getName(), "fleajpatest");
String sql = "select count(*) from student where stu_state = ?";
List<Object> paramList = new ArrayList<Object>(); paramList.add(1);
LOGGER.debug("COUNT = {}", FleaJDBCHelper.querySingle(sql, paramList)); }
|
执行结果:
3.3 JDBC更新数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Test public void testStudentUpdate() throws Exception { FleaJDBCConfig.init(DBSystemEnum.MySQL.getName(), "fleajpatest");
String sql = "update student set stu_state = ? where stu_name = ?";
List<Object> paramList = new ArrayList<Object>(); paramList.add(2); paramList.add("huazie");
int ret = FleaJDBCHelper.update(sql, paramList);
LOGGER.debug("RESULT = {}", ret); }
|
执行结果:
3.4 JDBC删除数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Test public void testStudentDelete() throws Exception { FleaJDBCConfig.init(DBSystemEnum.MySQL.getName(), "fleajpatest");
String sql = "delete from student where stu_name = ? and stu_state = ? ";
List<Object> paramList = new ArrayList<Object>(); paramList.add("huazie"); paramList.add(2);
int ret = FleaJDBCHelper.delete(sql, paramList);
LOGGER.debug("RESULT = {}", ret); }
|
执行结果: