从JDBC到手撸极简版Mybaties(3)JDBC自动解析配置文件

简介

开门见山的说,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();

运行效果:
在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值