简介
开门见山的说,java中有一个类叫Properties。该类主要用于读取Java的配置文件,不同的编程语言有自己所支持的配置文件,配置文件中很多变量是经常改变的,为了方便用户的配置,能让用户够脱离程序本身去修改相关的变量设置。就像在Java中,其配置文件常为.properties文件,是以键值对的形式进行参数配置的。
所以,我们可以尝试使用这个类来自动加载一个我们自定义的配置文件。然后,嘿嘿,我们的JDBC封装类就可以更方便一些了。
具体操作
首先,我们要先知道Properties的具体使用方法。emmmm,,网上代码一大堆,这里就直接给出来了:
在这里创建我们的配置文件:

然后代码如下:
Properties pro = new Properties();
pro.load(ReflectTest.class.getClassLoader().getResourceAsStream("pro.properties"));
String v = pro.getProperty("参数名");
代码解析
好了,那么问题来了,,,这几行代码干了什么?为啥就能找到了我们的配置文件?
首先,我们从反射开始讲起,,算了,太长了,这里直接说:反射可以实现是因为java中有个Class类,它的classLoader()可以把我们的字节码文件(.class)加载到内存中。。
等等,是不是发现了什么?嘿嘿,classLoader既然可以加载我们编译后的.class文件,是不是意味着它拥有我们编译后的文件路径?
来我们看看我们的target:

okk,可以发现我们的class文件下,除了我们的包,就是我们的配置文件了。当我们的classLoader加载一个.class文件时,我们是不是要输入完整的包名?那就是不是意味着,我们的classLoader的默认路径是我们的classes文件夹?
哈哈,这就破案了。
所以pro.load(ReflectTest.class.getClassLoader().getResourceAsStream("pro.properties"));
其中,ReflectTest.class.getClassLoader()就是在获取我们classLoader的路径,后面的函数顾名思义,就是获取字节流。而这个函数输入的内容,就是配置文件的相对classLoader的路径。
我们根据图可以知道,我们的配置文件就在classes下面,所以直接输入配置文件名字就可以啦。
合并代码
最后,把我们新功能合并到我们上篇的代码中:
package com.sy;
import java.io.IOException;
import java.lang.reflect.Field;
import java.sql.*;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
/**
* @author :byMJC
* @date :Created 2022/1/16 19:04
* @description:
*/
public class JDBCUtilPro {
private static final String DB_DEIVER = "com.mysql.jdbc.Driver";
private static String DB_URL ;
private static String DB_USER ;
private static String DB_PWD ;
private static Connection connection;
/**
* 静态代码块,注册驱动
*/
static {
try {
Class.forName(DB_DEIVER);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
/**
* 解析配置文件
* @param configName 文件名
* @throws IOException
*/
private void parseConfig(String configName) throws IOException {
Properties properties = new Properties();
properties.load(JDBCUtilPro.class.getClassLoader().getResourceAsStream(configName));
DB_URL = properties.getProperty("DB_URL");
DB_USER = properties.getProperty("DB_USER");
DB_PWD = properties.getProperty("DB_PWD");
}
/**
* 获取连接
*/
private void getConnection(){
try {
connection = DriverManager.getConnection(DB_URL, DB_USER, DB_PWD);
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
/**
* 构造器,传入配置文件名
* @param configName
*/
JDBCUtilPro(String configName){
try {
parseConfig(configName);
} catch (IOException e) {
e.printStackTrace();
}
getConnection();
}
/**
* 查询语句封装,返回值根据实体类成员顺序进行赋值
* @param sql sql语句
* @param args 参数,用List传
* @param tClass 返回值列表里的成员类型
* @return 返回查询结果的List
*/
public <T> List<T> selectResultByOrder(String sql, List<?> args, Class<T> tClass){
List<T> resList = new ArrayList<>();
int columLen;
try {
if (connection.isClosed()) {
getConnection();
}
// 利用反射获取属性
Field[] declaredFields = tClass.getDeclaredFields();
// 创建语句
PreparedStatement preparedStatement = connection.prepareStatement(sql);
// 赋值参数
for (int i = 0; i < args.size(); i++) {
preparedStatement.setObject(i+1, args.get(i));
}
// 执行语句
ResultSet resultSet = preparedStatement.executeQuery();
// 获取返回值的表数据和它的长度
ResultSetMetaData metaData = resultSet.getMetaData();
columLen = metaData.getColumnCount();
while (resultSet.next()) {
// 反射构建类
Object res = tClass.getDeclaredConstructor().newInstance();
// 遍历resultSet的每个属性值
for (int i = 0; i < columLen; i++) {
// 获取每一个返回对象的参数名
String columnName = metaData.getColumnName(i + 1);
// 获取对应名的参数
Object object = resultSet.getObject(columnName);
// 利用反射给res赋值参数
declaredFields[i].setAccessible(true);
declaredFields[i].set(res, object);
}
resList.add((T) res);
}
} catch (Exception e) {
e.printStackTrace();
return null;
}
return resList;
}
/**
* 查询语句封装,返回值根据实体类成员名称进行赋值
* @param sql sql语句
* @param args 参数,用List传
* @param tClass 返回值列表里的成员类型
* @return 返回查询结果的List
*/
public <T> List<T> selectResultByName(String sql, List<?> args, Class<T> tClass){
List<T> resList = new ArrayList<>();
try {
if (connection.isClosed()) {
getConnection();
}
// 利用反射获取属性
Field[] declaredFields = tClass.getDeclaredFields();
// 创建语句
PreparedStatement preparedStatement = connection.prepareStatement(sql);
// 赋值参数
for (int i = 0; i < args.size(); i++) {
preparedStatement.setObject(i+1, args.get(i));
}
// 执行语句
ResultSet resultSet = preparedStatement.executeQuery();
// 获取返回值的表数据和它的长度
ResultSetMetaData metaData = resultSet.getMetaData();
while (resultSet.next()) {
// 反射构建类
Object res = tClass.getDeclaredConstructor().newInstance();
for (Field declaredField : declaredFields) {
Object object = resultSet.getObject(declaredField.getName());
declaredField.setAccessible(true);
declaredField.set(res, object);
}
resList.add((T) res);
}
preparedStatement.close();
resultSet.close();
} catch (Exception e) {
e.printStackTrace();
return null;
}
return resList;
}
/**
* 关闭连接
*/
public void close(){
try {
connection.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
}
测试代码:
JDBCUtilPro jdbcUtilPro = new JDBCUtilPro("JDBCConfig.properties");
List<String> list = new ArrayList();
List<User1> list1 = jdbcUtilPro.selectResultByName("select * from user", list, User1.class);
System.out.println(list1);
jdbcUtilPro.close();
运行效果:












