自定义持久层框架(更新ing)

拉勾教育Java高薪班学习笔记

平常工作太忙,没有时间来系统学习,所以还是狠狠心报名了高薪班,希望系统的提升自己的技术,为了以后更好的发展.冲鸭!

1.JDBC回顾及问分析

Mybatis的出现是为了解决JDBC中的问题

我们与数据库的交互是通过JDBC,而Mybatis的出现就是为了帮我们解决在JDBC中出现的问题。

那么在原始的JDBC的交互中会出现什么问题呢?

如下所示是我们使用JDBC进行查询的代码

    public static void main(String[] args) {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        ResultSet resultSet = null;
        try {
            //加载数据库驱动
            Class.forName("com.mysql.jdbc.Driver");
            //通过驱动管理类获取数据库连接
            connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/mybatis?
                    characterEncoding = utf - 8", " root", " root");
                    //定义sql语句,?表示占位符
                    String sql = "select * from user where username = ?";
            // 获取预处理statement
            preparedStatement = connection.prepareStatement(sql);
            // 设置参数
            preparedStatement.setString(1, "tom");
            // 执行查询
            resultSet = preparedStatement.executeQuery();
            // 遍历查询结果
            while (resultSet.next()) {
                int id = resultSet.getInt("id");
                String username = resultSet.getString("username");
                // 封装数据
                user.setId(id);
                user.setUsername(username);
            }
            System.out.println(user);
        } catch (
                Exception e) {
            e.printStackTrace();
        } finally {
            // 释放资源
            if (resultSet != null) {
                try {
                    resultSet.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if (preparedStatement != null) {
                try {
                    preparedStatement.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if (connection != null) {
                try {
                    connection.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
        }
    }

我们不难看出其中的问题

问题

  • 硬编码

数据库的配置,sql语句的配置就直接写死在代码中,给我们后续的维护带来极大的麻烦。

  • 资源浪费

每次调用都会创建新的数据库连接,导致资源的极大浪费。

  • 手动封装返回结果,太繁琐

每次调用都需要手动将查询结果封装到java bean中。

解决思路

  • 配置文件

硬编码的问题一般都使用配置文件,或配置中心来解决,所以对应的我们需要将数据库配置信息和sql信息存放在配置文件中,并且因为sql信息会和不同的业务相关,而数据库信息一般不变,所以两者应该分别存放

  • 连接池

使用连接池来管理数据库连接,做到连接多用。

  • 使用反射,内省

使用反射,内省技术来映射需要装载的类。

2.自定义持久层框架思路分析

这就是思考mybatis的工作原理

为什么

我们要自定义持久层框架的目的是为了解决上一讲提出的问题。那么我们就应该按照上一讲的思路来思考。

做什么

我们的持久层框架是为了项目使用方能更加便捷的使用数据库,那么就要求可以使用jar的依赖引用来直接达到使用目的。

而上文提到的配置因为框架并不清楚,所以这部分的配置文件应该放在使用项目中维护,再交由框架读取使用。

怎么做

  1. 由项目创建两个配置文件

    1. sqlMapConfig:存放数据库配置信息,存放mapper.xml的全路径
    2. mapper.xml:存放sql信息
  2. 创建持久层框架(本质就是对JDBC操作进行了封装)

  3. 加载配置文件:根据配置文件的路径,将配置文件加载成字节流,存入内存中,等待使用

    1. 创建Rescources类,方法:getResourcesAsStream(String path)
  4. 创建两个javaBean:(容器对象):用来存放对加载好的字节流的解析出来的内容

    1. Configuration:核心配置类:存放sqlMapConfig解析出来的内容。
    2. MapperStatment:sql配置类:存放mapper解析出来的内容
  5. 使用dom4f来解析配置文件

    1. 创建sqlSessionFactoryBuilder类,方法:build(InputStream in)
    2. 使用dom4f来解析xml文件,并将解析结果存放在容器对象中。
    3. 创建sqlSessionFactory接口,用于生产sqlSession。(工厂模式)
  6. 创建sqlSessionFactory的默认实现类,DefaultSqlSessionFactory类,使用openSqlSession来生产sqlSession

  7. 使用上一步创建完成的sqlSession的默认实现类DefaultSqlSession来定义一个crud的操作

  8. 创建Executor接口及其实现类SimpleExecutor中的query(Configuration,MapperStatment,Object... param)来具体执行JDBC操作

  9. 将返回的数据使用反射和内省来装载。(视频中暂未提及)

补充

其实在使用dom4f来解析对象时应该也要有对应的专门的解析接口类.Resolver,并使用他的XMLResolver来解析

# 3.配置文件的代码实现
  • 创建业务项目

使用springboot快速创建即可

  • 创建sqlMapConfig文件
    • 数据库信息
    • mapper文件信息(为了加载时可以直接找到mapper文件的路径)
<configuration>
<!--    数据库配置信息-->
    <dataSource>
        <property name="driverClass" value="com.mysql.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql:///IPersistence"/>
        <property name="username" value="root"/>
        <property name="password" value="123456"/>
    </dataSource>
<!--    mapper配置信息-->
    <mapper resource="UserMapper.xml"></mapper>
</configuration>
  • 创建mapper文件
    • sql语句
    • sql语句对应爹statementId
    • mapper文件对应的命名空间(为了区分多个mapper文件)
    • 规定返回数据要反射到的数据类型
    • 规定查询数据中的参数对应的JAVA类型
    • 将sql语句中的?占位符使用特殊标签进行标记
<mapper namespace="user">
    <select id="selectList" resultType="com.huixiang.demo.user">
        select * from user
    </select>

    <select id="selectOne" resultType="com.huixiang.demo.user" paramterType="com.huixiang.demo.user">
        select * from user where id = #{} and username = #{}
    </select>
</mapper>

4.Resource类的定义

下面我们就开始按照我们持久层的步骤来实现.

创建Resource类

Resource类的作用很简单,就是根据路径来获取输入流.实际上可以使用约定的方式来约定配置文件的路径和文件名.但是在我们的项目中先不采用这种方式,而是通过客户端的传入路径来操作

package com.huixiang.io;

import java.io.InputStream;

/**
 * 资源类,用于获取输入流
 */
public class Resource {

    public static InputStream getInputStream(String path){
        return Resource.class.getClassLoader().getResourceAsStream(path);
    }
}

打包项目

然后将项目打包,由客户端应用引用即可并调用即可

package com.huixiang.test_app.test;

import com.huixiang.io.Resource;

import java.io.InputStream;

public class Test {
    public void test(){
        InputStream inputStream = Resource.getInputStream("sqlMapConfig.xml");
    }
}

5.容器对象的定义

这次要定义我们的容器对象,也就是要将字节流中解析的数据装载在容器对象中.

MapperStatement

这个是装载mapper文件的.那么对于mapper中的关键信息就都要存储.同时因为我们会有多个mapper,每个mapper中会有多个方法,所以我们对于MapperStatement一一对应了mapper中的每个方法.具体参见代码

package com.huixiang.pojo;

/**
 * mapper类
 * 重要的就是mapper中的各种信息的存储
 */
public class MapperStatement {
    /**
     * 对应mapper中的id
     */
    private String id;
    /**
     * 对应mapper中的返回类型
     */
    private String resultType;
    /**
     * 对一你个mapper中的参数类型
     */
    private String parameterType;
    /**
     * 对应sql语句
     */
    private String sql;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getResultType() {
        return resultType;
    }

    public void setResultType(String resultType) {
        this.resultType = resultType;
    }

    public String getParameterType() {
        return parameterType;
    }

    public void setParameterType(String parameterType) {
        this.parameterType = parameterType;
    }

    public String getSql() {
        return sql;
    }

    public void setSql(String sql) {
        this.sql = sql;
    }
}

Configuration

这个是装载数据库配置的,我们也可以直接将配置生成datasource对象来存储.同时因为真正在执行语句时还需要MapperStatement,所以可以直接将MapperStatement类装载在其中.

package com.huixiang.pojo;

import javax.sql.DataSource;
import java.util.HashMap;
import java.util.Map;

/**
 * 配置类
 * 已将流中的数据装载在配置类中.有数据库信息和mapper信息
 */
public class Configuration {

    private DataSource dataSource;
    /**
     * mapper类
     * key是statementId
     */
    private Map<String,MapperStatement> mapperStatementMap = new HashMap<>();

    public DataSource getDataSource() {
        return dataSource;
    }

    public void setDataSource(DataSource dataSource) {
        this.dataSource = dataSource;
    }

    public Map<String, MapperStatement> getMapperStatementMap() {
        return mapperStatementMap;
    }

    public void setMapperStatementMap(Map<String, MapperStatement> mapperStatementMap) {
        this.mapperStatementMap = mapperStatementMap;
    }
}

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