使用DbUtils对JDBC封装实现面向实体查询
直接上代码
package org.smart4j.chapter2.helper; import org.apache.commons.dbcp2.BasicDataSource;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import org.apache.commons.dbutils.handlers.MapListHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.smart4j.chapter2.util.CollectionUtil;
import org.smart4j.chapter2.util.PropsUtil; import java.sql.Connection;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Properties; /**
* DbUtils 提供对 JDBC 的轻量级封装
* 通过 DbUtils 提供的 QueryRunner 对象可以面向实体(Entity)进行查询。
* 它的原理是:执行 SQL 语句并返回一个 ResultSet,随后通过反射去创建并初始化实体对象。
* Created by zhengbinMac on 2017/3/25.
*/
public class DatabaseHelper {
private static final Logger LOGGER = LoggerFactory.getLogger(DatabaseHelper.class); // 线程连接
private static final ThreadLocal<Connection> CONNECTION_THREAD_LOCAL;
// DbUtils
private static final QueryRunner QUERY_RUNNER;
// 线程池
private static final BasicDataSource DATA_SOURCE; static {
CONNECTION_THREAD_LOCAL = new ThreadLocal<>();
QUERY_RUNNER = new QueryRunner(); Properties conf = PropsUtil.loadProps("config.properties");
// 设置连接池,使用 DBCP 来获取数据库连接
DATA_SOURCE = new BasicDataSource();
DATA_SOURCE.setDriverClassName(conf.getProperty("jdbc.driver"));
DATA_SOURCE.setUrl(conf.getProperty("jdbc.url"));
DATA_SOURCE.setUsername(conf.getProperty("jdbc.username"));
DATA_SOURCE.setPassword(conf.getProperty("jdbc.password"));
} /**
* 获取数据库连接
*/
public static Connection getConnection() {
// 首先从 ThreadLocal 中获取
Connection connection = CONNECTION_THREAD_LOCAL.get();
// 若不存在,则创建一个新的 Connection,并最终将其放入 ThreadLocal 中
if (connection == null) {
try {
connection = DATA_SOURCE.getConnection();
} catch (SQLException e) {
LOGGER.error("get connection failure", e);
} finally {
CONNECTION_THREAD_LOCAL.set(connection);
}
}
return connection;
} /**
* 批量查询实体
*/
public static <T> List<T> queryEntityList(Class<T> entityClass, String sql, Object... params) {
List<T> entityList = null;
try {
Connection conn = getConnection();
entityList = QUERY_RUNNER.query(conn, sql, new BeanListHandler<T>(entityClass), params);
} catch (SQLException e) {
LOGGER.error("query entity list failure", e);
}
return entityList;
} /**
* 查询单个实体
*/
public static <T> T queryEntity(Class<T> entityClass, String sql, Object... params) {
T entity = null;
try {
Connection conn = getConnection();
entity = QUERY_RUNNER.query(conn, sql, new BeanHandler<T>(entityClass), params);
} catch (SQLException e) {
LOGGER.error("query entity failure", e);
e.printStackTrace();
}
return entity;
} public static List<Map<String, Object>> executeQuery(String sql, Object... params) {
List<Map<String, Object>> result = null;
try {
Connection conn = getConnection();
result = QUERY_RUNNER.query(conn, sql, new MapListHandler(), params);
} catch (SQLException e) {
LOGGER.error("query entity failure", e);
e.printStackTrace();
}
return result;
} /**
* 执行更新语句(update、insert 和 delete)
*/
public static int executeUpdate(String sql, Object... params) {
int updateRows = 0;
try {
Connection conn = getConnection();
updateRows = QUERY_RUNNER.update(conn, sql, params);
} catch (SQLException e) {
LOGGER.error("execute update failure", e);
e.printStackTrace();
}
return updateRows;
} /**
* 插入实体
*/
public static <T> boolean insertEntity(Class<T> entityClass, Map<String, Object> fieldMap) {
if (CollectionUtil.isEmpty(fieldMap)) {
LOGGER.error("can not insert entity: fieldMap is empty");
return false;
}
String sql = "INSERT INTO " + getTableName(entityClass);
StringBuilder colums = new StringBuilder("(");
StringBuilder values = new StringBuilder("("); // 插入实体的字段名,和字段值的占位符
for (String colum : fieldMap.keySet()) {
colums.append(colum).append(", ");
values.append("?, ");
}
colums.replace(colums.lastIndexOf(", "), colums.length(), ")");
values.replace(values.lastIndexOf(", "), values.length(), ")");
sql += colums + " VALUES " + values; // 插入实体的值
Object[] params = fieldMap.values().toArray();
return executeUpdate(sql, params) == 1;
} /**
* 更新实体
*/
public static <T> boolean updateEntity(Class<T> entityClass, long id, Map<String, Object> fieldMap) {
if (CollectionUtil.isEmpty(fieldMap)) {
LOGGER.error("can not update entity: fieldMap is empty");
return false;
}
// 更具 fieldMap 拼接出更新 SQL 语句
String sql = "UPDATE " + getTableName(entityClass) + " SET ";
StringBuilder columns = new StringBuilder();
// 更新实体的字段
for (String colums : fieldMap.keySet()) {
columns.append(columns).append("=?, ");
} // 去掉 SQL 最后一个 ', '
sql += columns.substring(0, columns.lastIndexOf(", ")) + " WHERE id=?"; // 更新实体的值
List<Object> paramList = new ArrayList<Object>();
paramList.addAll(fieldMap.values());
paramList.add(id); // 增加主键 id
Object[] params = paramList.toArray(); return executeUpdate(sql, params) == 1;
} /**
* 删除实体
*/
public static <T> boolean deleteEntity(Class<T> entityClass, long id) {
String sql = "DELTE FROM " + getTableName(entityClass) + "WHERE id=?";
return executeUpdate(sql, id) == 1;
} /**
* 获取操作表的表名,即实体的类名
*/
private static String getTableName(Class<?> entityClass) {
return entityClass.getSimpleName();
}
}
两个工具类
集合工具类:
package org.smart4j.chapter2.util; import org.apache.commons.collections.MapUtils; import java.util.Collection;
import java.util.Map; /**
* 集合工具类
*
* Created by zhengbinMac on 2017/3/25.
*/
public class CollectionUtil {
/**
* 判断 Collection 是否为空
*/
public static boolean isEmpty(Collection<?> collection) {
return CollectionUtil.isEmpty(collection);
} /**
* 判断 Collection 是否为非空
*/
public static boolean isNotEmpty(Collection<?> collection) {
return !isEmpty(collection);
} /**
* 判断 Map 是否为空
*/
public static boolean isEmpty(Map<?, ?> map) {
return MapUtils.isEmpty(map);
} /**
* 判断 Map 是否为非空
*/
public static boolean isNotEmpty(Map<?, ?> map) {
return !isEmpty(map);
}
}
属性文件(properties)工具类:
package org.smart4j.chapter2.util; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties; /**
* 属性文件工具类
*
* Created by zhengbinMac on 2017/3/25.
*/
public class PropsUtil {
private static final Logger LOGGER = LoggerFactory.getLogger(PropsUtil.class); /**
* 加载属性文件
*/
public static Properties loadProps(String fileName) {
Properties properties = null;
InputStream is = null;
try {
is = Thread.currentThread().getContextClassLoader().getResourceAsStream(fileName);
if (is == null) {
throw new FileNotFoundException(fileName + "file is not found");
}
properties = new Properties();
properties.load(is);
} catch (IOException e) {
LOGGER.error("load properties file failure", e);
} finally {
if (is != null) {
try {
is.close();
} catch (IOException e) {
LOGGER.error("close inputStream failure", e);
}
}
}
return properties;
} /**
* 获取字符型属性(默认值为空字符串)
*/
public static String getString(Properties properties, String key) {
return getString(properties, key, "");
} /**
* 获取字符型属性(可指定默认值)
*/
public static String getString(Properties properties, String key, String defaultValue) {
String value = defaultValue;
if (properties.contains(key)) {
value = properties.getProperty(key);
}
return value;
} /**
* 获取数值型属性(默认值为0)
*/
public static int getInt(Properties properties, String key) {
return getInt(properties, key, 0);
} /**
* 获取数值型属性(可指定默认值)
*/
public static int getInt(Properties properties, String key, int defaultValue) {
int value = defaultValue;
if (properties.contains(key)) {
value = CastUtil.castInt(properties.getProperty(key));
}
return value;
} /**
* 获取布尔型属性(默认值为 false)
*/
public static boolean getBoolean(Properties properties, String key) {
return getBoolean(properties, key, false);
} /**
* 获取布尔型属性(可指定默认值)
*/
public static boolean getBoolean(Properties properties, String key, boolean defaultValue) {
boolean value = defaultValue;
if (properties.contains(key)) {
value = CastUtil.castBoolean(properties.getProperty(key));
}
return value;
}
}
参考资料
[1] 架构探险, 第 2 章 - 为 Web 应用添加业务功能
使用DbUtils对JDBC封装实现面向实体查询的更多相关文章
- Apache—dbutils开源JDBC工具类库简介
Apache—dbutils开源JDBC工具类库简介 一.前言 commons-dbutils 是 Apache 组织提供的一个开源 JDBC工具类库,它是对JDBC的简单封装,学习成本极低,并且使用 ...
- 优化JDBC封装
可重用性较强的JDBC封装 以下为代码,注释中写了主要思想 主类 com.util.JDBCUtil.java package com.util; import java.lang.reflect.F ...
- jdbc封装代码
jdbc封装代码 package jdbcUtil; import java.sql.Connection; import java.sql.DriverManager; import java.sq ...
- MyBatis Plus 将查询结果封装到指定实体
MyBatis Plus 将查询结果封装到指定实体 思路 自定义方法,使用Wrapper,自定义映射结果集 Mapper接口 package com.mozq.boot.mpsand01.dao; i ...
- Phoenix的jdbc封装
一.Phoenix版本 <dependency> <groupId>org.apache.phoenix</groupId> <artifactId>p ...
- JDBC封装的工具类
1. JDBC封装的工具类 public class JDBCUtil { private static Properties p = new Properties(); private static ...
- Java -- JDBC_利用反射及 JDBC 元数据编写通用的查询方法
先利用 SQL 进行查询,得到结果集: 利用反射创建实体类的对象:创建对象: 获取结果集的列的别名: 再获取结果集的每一列的值, 结合 3 得到一个 Map,键:列的别名,值:列的值: 再利用反射为 ...
- 通过Java反射做实体查询
我们在使用hibernate的时候,查询的时候都会和实体中的一些字段相结合去查询,当然字段少了,还算是比较简单,当字段多了,就不那么容易了,所以就自己写了个方法,根据实体中的字段信息去查询,废话不多说 ...
- JDBC_14_使用JDBC工具类实现模糊查询
使用JDBC工具类实现模糊查询 代码: import java.sql.*; /** * 模糊查询 * 测试DBUtils */ public class JDBCTest09 { public st ...
随机推荐
- WebAPI调用笔记 ASP.NET CORE 学习之自定义异常处理 MySQL数据库查询优化建议 .NET操作XML文件之泛型集合的序列化与反序列化 Asp.Net Core 轻松学-多线程之Task快速上手 Asp.Net Core 轻松学-多线程之Task(补充)
WebAPI调用笔记 前言 即时通信项目中初次调用OA接口遇到了一些问题,因为本人从业后几乎一直做CS端项目,一个简单的WebAPI调用居然浪费了不少时间,特此记录. 接口描述 首先说明一下,基于 ...
- xtrabackup 源码安装
安装依赖包:这些依赖包必须要先安装好 # yum install cmake libaio-devel ncurses-devel bzip2-devel libxml2-devel libgcryp ...
- 【转】Tesla Autopilot
Tesla Autopilot 以下内容是<Tesla Model S的设计失误>一文中新加入的小节.由于写作时间相距太远,而且由于它的时效性,现在也把它单独提出来,独立成文. 两个月前, ...
- Android中使用IntentService运行后台任务
IntentService提供了一种在后台线程中运行任务的方式,适合处理运行时间较长的后台任务. 长处: (1)IntentService执行在单独的线程中.不会堵塞UI线程 (2)IntentSer ...
- C#基础第四天-作业-Hashtable-list<KeyValuePair>泛型实现名片
1.名片集: 名片集实现功能:不限定集合实现 增加,查询,查询全部,删除 功能,需求条件: 根据姓名去(查询/删除)某一行数据.指定:姓名,年龄,性别,爱好,联系方式. 采用技术:Hashtable- ...
- java字符串的遍历以及字符串中各类字符的统计
1.需求:获取字符串中的每一个字符 分析: A:如何能够拿到每一个字符呢? char charAt(int index) B:我怎么知道字符到底有多少个呢? int length() publi ...
- 用Entityframework 调用Mysql时,datetime格式插入不进去数据库的解决办法。
1. 打开Model.edmx, 2. 选择userinfo中的createtime字段的属性 3. storegeneratedpattern设置值为None
- 一篇文章学会shell工具篇之sed
sed工具执行原理; 有关sed的参数及action的常见操作方法; 定址; 模式空间和保持空间; 使用标签 1.首先先来了解一下什么是sed? sed叫做流编辑器,在shell脚本和Makefile ...
- vue.js $refs和$emit 父子组件交互
父调子 $refs (把父组件的数据传给子组件) <template> <div id="app"> <input type="butto ...
- mongoose修改数组中某个特定的值
写博客的时候有一个这样的业务,一个标签集合和一个文章集合,它们是多对多的关系,文章集合中tags字段包含它对应的标签,现在修改标签集合中某条标签记录的名字,文章集合中所有包含这个标签的tags字段的值 ...