设计模式学习笔记 - 开源实战五(上):Mybatis如何权衡易用性、性能和灵活性?

本文主要是介绍设计模式学习笔记 - 开源实战五(上):Mybatis如何权衡易用性、性能和灵活性?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

前面几篇文章讲解了 Spring 框架,剖析了背后蕴含的一些通用设计思想、设计原则,以及用到的十几种设计模式。从本章开始,我们再剖析一个 Java 项目中经常用到的开发框架:Mybatis。本次也分为三篇文章进行讲解:

  • 第一篇文章,分析 Mybatis 如何权衡代码的易用性、性能和灵活性。
  • 第二篇文章,学习如何利用职责链模式与代理模式实现 Mybatis Plugin。
  • 第三篇文章,总结逻辑下 Mybatis 框架中用到的十几种设计模式。

Mybatis 和 ORM 框架

熟悉 Java 的同学应该知道,Mybatis 是一个 ORM(Object Relational Mapping,对象 - 关系映射)框架。ORM 框架主要是根据类和数据库之间的映射关系,帮助程序员自动实现对象与数据库中数据之间的互相转化。说的更具体点就是,ORM 负责将程序中的对象存储到数据库中、将数据库中的数据转化为程序中的对象。实际上,Java 中的 ORM 框架有很多,除了 Mybatis 之外,还有 Hibernate、TopLink 等。

在剖析 Spring 框架时,我们讲到,如果用一句来总结框架的作用,那就是简化开发。Mybatis 框架也不例外。它简化的是数据库方面的开发。Mybatis 是如何简化数据库开发的呢?

我们结合模板模式(下)章节 中将的 JdbcTemplate 的例子来说明下。

在《模板模式(下)》中,我们讲到,Java 提供了 JDBC 类库来封装不同类型的数据库操作。不过,直接使用 JDBC 来进行数据库编程,还是有点麻烦。语句,Spring 提供了 JdbcTemplate,对 JDBC 进一步封装,来进一步简化数据库编程。使用 JdbcTemplate 进行数据库编程,只需要编写和业务相关的代码(比如,SQL 语句、数据库中数据与对象之间的互相转化的代码),其他流程性质的代码(比如,加载驱动、创建数据库连接、创建 statement、关闭连接、关闭 statement 等)都封装在了 JdbcTemplate 中,不需要我们重复编写。

当时为了展示使用 JdbcTemplate 是如何简化数据库编程的,还举了一个查询数据库中用户信息的例子。还是同样的例子,再来看下使用 Mybatis 该如何实现,是不是比 JdbcTemplate 更加简单。

因为 Mybatis 依赖 JDBC 驱动,所以,在项目中使用 Mybatis,除了需要引入 Mybatis 框架本身(Mybatis.jar)之外,还需要引入 JDBC 驱动(比如,访问 MYSQL 的 JDBC 驱动实现类 mysql-connector-java.jar)。两个 jar 包引入项目之后,就可以开始编程了。使用 Mybatis 来访问数据库中用户信息的代码如下所示。

// 1.定义UserDO
public class UserDo {private long id;private String name;private String telephone;// 省略setter/getter方法
}// 2.定义访问接口
public interface UserMapper {public UserDo selectById(long id);
}// 3.定义映射关系:UserMapper.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org/DTD Mapper 3.0//EN"       "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
<mapper namespace="com.example.repo.mapper.UserMapper"><select id="selectById" resultType="com.example.repo.UserDo">select * from user where id=#{id}</select>
</mapper>// 4.全局配置文件:mybatis.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org/DTD Mapper 3.0//EN"       "http://mybatis.org/dtd/mybatis-3-config.dtd" >
<configuration><environments default="dev"><environment id="dev"><transactionManager type="JDBC"></transactionManager><dataSource type="POOLED"><property name="driver" value="com.mysql.jdbc.Driver" /><property name="url" value="jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=UTF-8" /><property name="username" value="root" /><property name="password" value="..." /></dataSource></environment></environments><mappers><mapper resource="mapper/UserMapper.xml"/></mappers>
</configuration>

需要注意的是,在 UserMapper.xml 配置文件中,我们之定义了接口和 SQL 语句之间的映射关系,并没有显式地定义类(UserDo)字段与数据库表(user)之间的映射关系。实际上,这就体现了 “约定优先于配置” 的设计原则。类与数据库表字段之间使用了默认的映射关系:类字段跟数据库表中拼写相同的字段一一映射。当然,如果没有办法做到一一映射,我们也可以自定义它们之间的映射关系。

有了上面的代码和配置,我们就可以向下面这样来访问数据库中的用户信息了。

public class MybatisDemo {public static void main(String[] args) throws IOException {Reader reader = Resources.getResourceAsReader("mybatis.xml");SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(reader);SqlSession session = sessionFactory.openSession();UserMapper userMapper = session.getMapper(UserMapper.class);UserDo userDo = userMapper.selectById(8);// ...}
}

从代码中可以看出,相对于使用 JdbcTemplate 的实现方式,使用 Mybatis 的实现方式更加灵活。在使用 JdbcTemplate 的实现方式中,对象与数据库之间的转化代码、SQL 语句,是硬编码在业务代码中。而使用 Mybatis 的实现方式中,类字段和数据库字段之间的映射关系、接口与 SQL 之间的映射关系,是写在 XML 配置文件中的,是跟代码相分离的,这样会更加灵活、清晰,维护起来更加方便。

如何平衡易用性、性能和灵活性?

刚刚对 Mybatis 做了简单的介绍,接下来,再对比下另外两个框架:JdbcTemplate 和 Hibernate。通过对比来看,Mybatis 是如何权衡代码的易用性、性能和灵活性的。

先来看 JdbcTemplate。相对于 Mybatis 来说,JdbcTemplate 更加轻量级。因为它对 JDBC 只做了很简单的封装,所以性能损耗比较少。相对于其他两个框架来说,它性能最好。但是,它的缺点也比较明显,那就是 SQL 与代码耦合在一起,而且不具备 ORM 功能,需要自己编写代码,解析对象跟数据库之间的映射关系。所以,在易用性上它不及其他两个框架。

再来看 Hibernate。相对 Mybatis 来说,Hibernate 更加重量级。Hibernate 提供了更加高级的映射功能,能够根据业务需求自动生成 SQL 语句。我们需要像使用 Mybatis 那样自己编写 SQL。因此,有时候,我们也把 Mybatis 称半自动化的 ORM 框架,把 Hibernate 称全自动化的 ORM 框架。不过,虽然自动生成 SQL 简化了开发,但毕竟是自动生成的,没有针对性能的优化。在性能方面,这样得到的 SQL 可能没有程序员编写得好。同时,这样也丧失了灵活性。

所以,粗略地讲,有时候框架的易用性和性能成对立关系。追求易用性,那性能就差一些。相反,追求性能,易用性就差一些。此外,使用起来越简单,那灵活性就越差。这就好比我们用照相机。傻瓜相机按下快门就能拍照,但没有复杂的单反灵活。

实际上,JdbcTemplate、Mybatis、Hibernate 这几个框架之间也体现了刚刚的说的规律。

JdbcTemplate 提供的功能最简单,易用性最差,性能损耗最少,用它编程性能最好。Hibernate 提供的功能最完善,易用性最好,但相对来说性能损耗就高了。Mybatis 介于两种中间,在易用性、性能、灵活性三个方面做了权衡。它支持程序员自己编写 SQL,能够延续程序员对 SQL 知识的积累。相对于完全黑盒子的 JdbcTemplate,很多程序员反倒是更加喜欢 Mybatis 这种半透明的框架。这也提醒我们,过度封装,提供过于简化的开发方式,也是丧失开发的灵活性。

总结

JDBC 是 Java 访问数据库的开发规范,提供了一套抽象的统一的开发接口,隐藏不同数据库的访问细节。

JdbcTemplate、Mybatis、Hibernate 都是对 JDBC 的二次封装,为的是进一步简化数据库开发。其中 JdbcTemplate 不能算得上是 ORM 框架,因为还需要程序员自己编程来实现对象和数据库之间的相互转化。相对于 Hibernate 这种连 SQL 都不用程序员写的全自动 ORM 框架,Mybatis 算式一种半自动化的 ORM 框架。

本章除了介绍 Mybatis 背景之外,还学习了代码的易用性、性能、灵活性之间的关系。一般来讲,提供的高级功能越多,那性能损耗就越大;用起来越简单,提供越简化的开发方式,那灵活性也就相对越低。

这篇关于设计模式学习笔记 - 开源实战五(上):Mybatis如何权衡易用性、性能和灵活性?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/934519

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

mybatis的整体架构

mybatis的整体架构分为三层: 1.基础支持层 该层包括:数据源模块、事务管理模块、缓存模块、Binding模块、反射模块、类型转换模块、日志模块、资源加载模块、解析器模块 2.核心处理层 该层包括:配置解析、参数映射、SQL解析、SQL执行、结果集映射、插件 3.接口层 该层包括:SqlSession 基础支持层 该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

阿里开源语音识别SenseVoiceWindows环境部署

SenseVoice介绍 SenseVoice 专注于高精度多语言语音识别、情感辨识和音频事件检测多语言识别: 采用超过 40 万小时数据训练,支持超过 50 种语言,识别效果上优于 Whisper 模型。富文本识别:具备优秀的情感识别,能够在测试数据上达到和超过目前最佳情感识别模型的效果。支持声音事件检测能力,支持音乐、掌声、笑声、哭声、咳嗽、喷嚏等多种常见人机交互事件进行检测。高效推