mybatis(1)谈谈JDBC

一、JDBC!你会用么?

(一)什么是JDBC

在讲解mybatis之前,先了解下JDBC

Java数据库连接,(Java Database Connectivity,简称JDBC)是java语言中用来规范客户端程序如何来访问数据库的

应用程序接口,提供了诸如查询和更新数据库中数据的方法。(摘自百度百科

JDBC是Java语言访问数据库的规范,Java操作数据库离不开JDBC。类似hibernate和mybatis也都是在JDBC的基础上进行了进一步封装,将复杂和重复的工作抽取出来,从而简化JDBC代码开发,使开发者更加专注于业务逻辑的开发,可以说JDBC是一切Java 持久层框架的基石。

(二)JDBC的使用

现在想一想在不使用spring、mybatis等其他第三方提供的框架时,如何实现一个简单的查询功能?

作者在此给出参考:

/**
 * 以mysql为例,读者可以自行建库,编写语句,此代码只做参考。
 */
public class JdbcDemo {
    public static void main(String[] args) throws Exception {
         //1.数据库配置信息,当然这些信息在实际开发中肯定要写在配置文件中的,此处仅做演示使用
        String url = "jdbc:mysql://127.0.0.1:3306/mybatis_study";
        String driverClass = "com.mysql.jdbc.Driver";
        String username = "root";
        String password = "123456";
        //2.获取连接,首先根据驱动类加载驱动
        Class.forName(driverClass);
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        ResultSet resultSet = null;
        try {
            connection = DriverManager.getConnection(url, username, password);
            //3.执行查询
            String sql = "SELECT * FROM t_user";
            preparedStatement = connection.prepareStatement(sql);
            resultSet = preparedStatement.executeQuery();
            //4.遍历结果集,并封装对象
            List<User> users = new ArrayList<>();
            while (resultSet.next()) {
                int id = resultSet.getInt("user_id");
                String userName = resultSet.getString("user_name");
                int userAge = resultSet.getInt("user_age");
                Date createDate = resultSet.getDate("create_date");
                User user = new User();
                user.setUserId(id);
                user.setUserName(userName);
                user.setUserAge(userAge);
                user.setCreateDate(createDate);
                users.add(user);
            }
            System.out.println(users);
            //5.释放资源
        } finally {
            try{
                if (resultSet != null) {
                    resultSet.close();
                }
                if (preparedStatement != null) {
                    preparedStatement.close();
                }
                 if (connection != null) {
                    connection.close();
                }
            }catch(Exception e){
                e.printStackTrace();
            }
        }
    }

如上就是一个简单的查询数据功能的实现,你是否能完整的写出来,是否能知道每步的操作?

java连接数据库并执行操作的基本步骤:

  1. 加载JDBC驱动程序

  2. 获得数据库连接

  3. 获得执行sql的对象

  4. 执行sql并获得结果集

  5. 遍历结果集,封装成java对象

  6. 关闭JDBC连接,释放资源

这里只是简单的JDBC使用,例如事务处理,存储过程调用等操作,也完全是基于JDBC实现的,可以说JDBC基本包含了数据库的所有操作,不懂JDBC的话很难真正了解mybatis。

(三)直接使用JDBC存在的问题

如(二)中的代码,一个简单的查询功能,使用JDBC来实现时,我们需要写好多的代码。毫不夸张的说,这些代码有时候会比业务代码都要多,而且很多代码都是重复的,例如加载驱动、获得连接、释放连接等。

当然我们可以写一个工具类,例如这样:

public class JdbcHelper {

    //这些信息应该从配置文件中读取,此处直接写死,读者可以自行实现
    private static final String DRIVER_CLASS_NAME;
    private static final String URL;
    private static final String USERNAME;
    private static final String PASSWORD;

    static {
        
        DRIVER_CLASS_NAME = "com.mysql.jdbc.Driver";
        URL = "jdbc:mysql://127.0.0.1:3306/mybatis_study";
        USERNAME = "root";
        PASSWORD = "123456";
        
        try {
            Class.forName(DRIVER_CLASS_NAME);
        } catch (ClassNotFoundException e) {
            // 这里需要日志输出,用控制台输出代替,下同
            System.out.println("驱动加载失败:" + e.getMessage());
        }
    }

    /**
     * 获得数据库连接
     *
     * @return 数据库连接对象
     */
    public static Connection getConnection() {
        Connection connection = null;
        try {
            connection = DriverManager.getConnection(URL, USERNAME, PASSWORD);
        } catch (SQLException e) {
            System.out.println("数据库连接获取失败:" + e.getMessage());
        }
        return connection;
    }


    /**
     * 关闭数据库连接
     *
     * @param connection 数据库连接对象
     */
    public static void closeConnection(Connection connection) {
        if (connection != null) {
            try {
                connection.close();
            } catch (SQLException e) {
                System.out.println("数据库连接关闭失败:" + e.getMessage());
            }
        }
    }

    /**
     * 执行查询,并返回列表集合
     *
     * @param entityeClass 泛型类型
     * @param sql          要执行查询的sql语句
     * @param params       查询参数
     * @return 查询结果
     */
    public static <T> List<T> queryEntityList(Class<T> entityeClass, String sql, Object... params) {

        List<T> list = new ArrayList<>();
        Connection connection = getConnection();
        try {
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            //此处没考虑动态SQL
            for (int i = 0; i < params.length; i++) {
                preparedStatement.setObject(i + 1, params[i]);
            }
            ResultSet resultSet = preparedStatement.executeQuery();
            
            //.....利用反射等技术封装entityClass对象,并添加到list

        } catch (SQLException e) {
            System.out.println(e.getMessage());
        } finally {
            closeConnection(connection);
        }
        return list;
    }
    ....其他方法不再累述,可自行实现
}

这样我们可以解决大量JDBC操作的代码,但这就完美了么?

首先,sql语句与java代码还是耦合在一起的;

其次,结果集的封装比较复杂的 (这个读者可以试着写一下

再次,动态sql编写的逻辑复杂(这个读者可以试着写一下

其他 ...

动态SQL

在编译时无法确定,只有等到程序运行起来,在执行的过程中才能确定,这种SQL叫做动态SQL。

例如这样一个场景,有个用户界面(用户表t_user),界面上有一系列查询条件例如“用户名查询(user_name)”、“所属分组(user_group)”、“性别查询(user_sex)”等,当然这些查询条件都是非必填选项。此时就会产生动态SQL的问题:

首先给出没有输入查询条件时的sql:SELECT * FROM t_user LIMIT ?,?

当输入用户名称查询时的sql :SELECT * FROM t_user WHERE user_name = ? LIMIT ?,?;

当输入用户名及性别时的sql :SELECT * FROM t_user WHERE user_name = ? AND user_sex = ? LIMIT ?,?

通过上述三个sql,用户输入的查询条件是随机的,导致要执行的sql是无法在编写代码阶段确定的,这就需要在执行查询时根据查询条件拼装相应sql,这就需要写一些逻辑代码来实现sql拼装;由于是sql是随机拼装的,导致查询参数也是随机的,也需要逻辑代码去设置...是不是很麻烦,就一个简单的查询业务,却在JDBC上花费了很多时间。

读者可以尝试实现下这个简单的场景,体会下用原生JDBC面临的问题。这样才能更好的理解我们在开发中使用mybatis的意义,以及mybatis为我们做了什么,还有mybatis各种配置的意义何在。正所谓知其然,更要知其所以然,多思考,多实践,多质疑,是一个程序员进步的关键。

综上,使用原生的jdbc来做数据访问存在着诸多不便,而且很多模板化的代码完全可以提取出来。而mybatis、hibernate等一系列持久层框架的诞生,正是为解决这些问题。接下来我们通过剖析mybatis的运行机制和部分源码,来学习下mybatis是如何工作的以及底层是如何封装JDBC的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,496评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,407评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,632评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,180评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,198评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,165评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,052评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,910评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,324评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,542评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,711评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,424评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,017评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,668评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,823评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,722评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,611评论 2 353

推荐阅读更多精彩内容