百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

5年码农的我整理了MyBatis从入门到实战,想精通看这一篇就够了

ztj100 2025-01-07 17:22 14 浏览 0 评论

今日分享开始啦,请大家多多指教~

mybatis简介

1 mybatis是什么

  • MyBatis 是一款优秀的持久层框架
  • MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集的过程
  • MyBatis 可以使用简单的 XML 或注解来配置和映射原生信息,将接口和 Java 的 实体类 【Plain Old Java Objects,普通的 Java对象】映射成数据库中的记录。

2 持久化和持久层

持久化

  • 将原本存储在内存的临时数据保存到可永久化的存储设备中。
  • JDBC就是一种持久化机制。文件IO也是一种持久化机制。
  • 持久化的需要是因为内存的缺陷引起的

持久层

  • 完成持久化工作的代码块 . ----> dao层 【DAO (Data Access Object) 数据访问对象】

3 Mybatis的优点

  • 使sql与程序代码解耦合
  • 灵活,不会对应用程序产生影响

mybatis快速构建

为了方便使用可以安装lombox插件和Free Mybatis plugn插件

流程:pom环境–>mybatis-configxml配置文件–>工具类–>实体类–>接口–>接口实现类–>测试类

1.pom.xml

2.mybatis-config.xml

3.SqlSessionUtil工具类(用于获取session)

4.实体类UserInf

5.接口UserMapper

6.List<UserInf> selectUser();

接口实现UserMapper.xml

别忘了在Mybatis-config.xml中注册

7.测试

mybatis的增删改查

session.commit(); //提交事务,重点!不写的话不会提交到数据库

1 namespace说明

namespace命名空间作为在mapper中的属性之一,是Mapper接口与xml实现的唯一依据。

2 select

UserMapper接口增加方法

//根据id查询用户

UserInf selectUserById(@Param("id")int id);

UserMapper.xml增加方法对应映射

测试

3 insert

UserMapper接口增加方法

//增加用户信息

int addUserInf(UserInf userInf);

UserMapper.xml增加方法对应映射

测试

4 update

UserMapper接口增加方法

//修改用户信息

int updateUserInf(UserInf userInf);

UserMapper.xml增加方法对应映射

测试

5 delete

UserMapper接口增加方法

//根据id删除用户

int deleteUser(@Param("id")int id);

UserMapper.xml增加方法对应映射

测试

6 模糊查询

第1种:在Java代码中添加sql通配符。

第2种:在sql语句中拼接通配符,会引起sql注入

7 注意点

所有的增删改操作都需要提交事务!

接口所有的普通参数,尽量都写上@Param参数,尤其是多个参数时,必须写上!

有时候根据业务的需求,可以考虑使用map传递参数!

为了规范操作,在SQL的配置文件中,我们尽量将Parameter参数和resultType都写上!

mybatis配置详解

1 核心配置文件

  • mybatis-confifig.xml 系统核心配置文件
  • MyBatis 的配置文件包含了会深深影响 MyBatis 行为的设置和属性信息
  • 一些属性和配置内容说明:

configuration(配置)

properties(属性)

settings(设置)

? typeAliases(类型别名)

? typeHandlers(类型处理器)

? objectFactory(对象工厂)

? plugins(插件)

? environments(环境配置)

? environment(环境变量)

? transactionManager(事务管理器)

? dataSource(数据源)

? databaseIdProvider(数据库厂商标识)

? mappers(映射器)

2 environments(环境配置)

根据生产上的要求,不同部门可能需要不同环境,可以配置多套环境,但是只能同时使用一个,且必须指定其中一个为默认运行环境(通过default指定)

子元素节点:environment

具体的一套环境,通过设置id进行区别,id保证唯一!

  • 子元素节点:transactionManager - [ 事务管理器 ]

MyBatis 中有两种类型的事务管理器(也就是 type="[JDBC|MANAGED]")在单独的mybatis中不需要进行额外配置

  • 子元素节点:dataSource(数据源)

dataSource 元素使用标准的 JDBC 数据源接口来配置 JDBC 连接对象的资源。

有三种内建的数据源类型(type=“[UNPOOLED|POOLED|JNDI]"))

3 mappers(映射器)

映射器 : 定义映射SQL语句文件

3.1 引入资源的不同方式

3.2 Mapper文件模板

4 properties(属性)

  • properties 属性可以利用配置文件或者java代码来动态更改
  • 在多处中配置了属性时,propertie中的值会被resouce中的覆盖,会被java代码中的方法传参覆盖。
  • 可以采用占位符的形式来配置属性默认值官方文档

4.1使用配置文件优化

新建config.properties

导入配置文件并占位符显示属性值

5 typeAliases(别名)

为解决包名过长问题的代码冗余。

第一种方式

在所有使用com.lyj.entity.UserInf的地方都可以用User代替

第二种方式

在com.lyj.entity包下的所有实体类都可以使用类名首字母小写的方式引用(如com.lyj.entity.UserInf用userInf代替)

也可以在实体类加上注解的方式自定义别名

@Alias("user")

public class UserInf { ... }

6 settings(设置)

可以为mybatis增加设置

数据库驼峰名转换设置

log日志设置显示

<setting name="logImpl" value="STDOUT_LOGGING"/>

ResultMap(结果集映射)

为了解决数据映射问题而生

1 解决数据库与实体类属性名不一致问题

解决的问题:属性名和字段名不一致

因为各种原因,数据库的属性字段无法和实体类的属性字段一一对应。

举例:

现在数据库表属性为:

实体类属性值为:

查询语句:

这样查询,因为pwd在实体类中并不存在,在注入值时无法注入。出现password属性为空的情况。

解决方案

使用别名(可简写 pwd password)

使用结果集映射->ResultMap 【推荐】

2 解决多对一结果集映射(association关联属性)

多对一理解:

  • 多个学生对应一个老师。
  • 数据库概念理解:学生拥有一个tid与教师id对应。
  • 实体类概念理解:学生拥有一个教师对象。

2.1 按查询嵌套处理

实体类student和teacher

StudentMapper增加接口

//查询所有学生

public List<Student> queryAllStudents();

StudentMapper.xml增加对应方法(对于对象属性teacher采用关联属性)

测试(对于对象属性teacher的属性采用关联属性)

2.2 按结果嵌套处理。

3解决一对多结果集映射(collection集合)

一对多理解:

  • 一个老师有多个学生
  • 数据库概念理解:教师拥有一个id与学生tid对应。
  • 实体类概念理解:教师拥有一个学生对象List集合。

3.1 按查询嵌套处理。

实体类Student和Teacher

TeacherMapper增加方法

public Teacher getTeacher(int id);

TeacherMapper.xml增加实现

测试

3.2 按结果嵌套处理

动态SQL

动态SQL指的是根据不同的查询条件 , 生成不同的Sql语句。

1 环境

utils工具类:随机生成id

entity实体类

其他正常布置,数据库添加数据。

2 if语句

需求:根据作者名字和博客名字来查询博客!如果作者名字为空,那么只根据博客名字查询,反之,则根据作者名来查询

xml文件实现

注意:在依次执行所有的if语句,因此当title为空,author不为空时,产生错误。引入where

3 where语句

xml文件实现

注意:当where包含的标签有返回值时能够自动加上where字符,同时当检测标签开头含有 AND或 OR 字符时,会将字符除去。

4 choose语句

利用if 和where搭配已经能够解决大部分的SQL查询拼接,但是有时候我们需要某一个条件不为空时立即跳出不在接受参数。这个时候我们可以利用choose实现,相当于编程语言中的 switch。

xml实现:

读起来十分有switch-case-otherwise的既视感。

5 foreach语句

foreach 相当于for循环,当我们需要一次性查询多个内容时,可以用foreach来实现这一功能。

测试:

相当于SQL语句:select * from blog where (id=1 or id=2 or id=3)

6 set语句

当我们需要进行修改操作时,set的操作配合if使用会现之前查询出现的问题。

不用set标签之前,当author为空时,名=SQL语句拼接就会出错,这个时候使用set标签可以很好的解决这个问题。

使用set标签后

**set标签的功能和where类似,先是内部标签有返回值时会自动补充set字符,当内部标签结尾含有,字符时,会自动去除 **

7 SQL片段

在正常的数据查询流程中,直接使用select *对数据安全来说无疑是十分危险的,我们会使用字段的形式来代替这种方式,但是字段过多的情况下,每次重复的复制代码过于麻烦且不容易维护。这个时候自定义SQL片段可以很好的解决这个问题。

sql标签和incloud标签配合使用很好的实现了这个功能。

日志工厂与分页

1 日志工厂

mybatis可以通过配置日志的方式来过程信息,可以采用自带的日志工厂,也可以用第三方包。

Mybatis内置的日志工厂提供日志功能,具体的日志实现有以下几种工具:

  • SLF4J
  • Apache Commons Logging
  • Log4j 2
  • Log4j
  • JDK logging

标准日志实现:

2 Log4j

通过使用Log4j,我们可以控制日志信息输送的目的地:控制台,文本,GUI组件…

我们也可以控制每一条日志的输出格式;

使用:

导包

更改日志工厂设置

新建配置文件log4j.properties

使用

**注意:**log4j导包使用的是apache的包,会自动在当前目录生成一个log文件夹,可以显示log记录。

3 limit实现分页

**分页:**记得以前一个javaweb项目当中,老师要求使用java代码来实习分页查询。那个时候是使用了一个工具类来专门处理分页逻辑,现在回顾过去,也不是十分困难。

主要有两个方法,一是在MySQL层次上处理,另一个是在java代码层次上。

3.1 分页实现

mysql语法:

步骤:

SQL语句

如果我们有很多种不同的分页类型,为每一个不同的类型写不同的工具类就会让代码十分的冗余,用泛型可以解决这一问题。

blog实体类

PageModel泛型类,来实现一个基本的分页数据。

如果想要在PageModel的基础上加上具有不同分页特色的数据,那么可以用继承来实现。

工具类处理

测试类

之前以为这样就完成了分页,没想到始终无法将Mapper抽象出来,这个分页是不合格的。

3.2 RowBounds分页

使用方法

3.3 PageHelper

使用方法

mybatis缓存

1 简介

什么是缓存 [ Cache ]?

  • 存在内存中的临时数据。
  • 将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题。

为什么使用缓存?

  • 减少和数据库的交互次数,减少系统开销,提高系统效率。

什么样的数据能使用缓存?

  • 经常查询并且不经常改变的数据。

2 Mybatis缓存

MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大地提升查询效率。

MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存

  • 默认情况下,只有一级缓存开启。(SqlSession级别的缓存,也称为本地缓存)
  • 二级缓存需要手动开启和配置,它是基于namespace级别的缓存。
  • 为了提高扩展性,MyBatis定义了缓存接口Cache。我们可以通过实现Cache接口来自定义二级缓存

3 一级缓存

一级缓存也称为本地缓存。

默认开启的。

SqlSession级别的缓存。

在同一个session中我们执行同一个查询,将不会重复的去sql读取,而是从缓存中去读取。

3.1 一级缓存失效的四种情况

  • sqlSession不同
  • sqlSession相同,查询条件不同
  • sqlSession相同,两次查询之间执行了增删改操作!
  • sqlSession相同,手动清除一级缓存。

session.clearCache();//手动清除缓存

4 二级缓存

二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存

基于namespace级别的缓存,一个名称空间,对应一个二级缓存;

工作机制

  • 一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中;
  • 如果当前会话关闭了,这个会话对应的一级缓存就没了;但是我们想要的是,会话关闭了,一 级缓存中的数据被保存到二级缓存中;
  • 新的会话查询信息,就可以从二级缓存中获取内容。
  • 不同的mapper查出的数据会放在自己对应的缓存(map)中;

4.1 二级缓存的使用

mybatis-config.xml开启全局缓存

在要开启缓存的namespace缓存配置官方文档

<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>

#这个更高级的配置创建了一个 FIFO 缓存,每隔 60 秒刷新,最多可以存储结果对象或列表的 512 个引用,而且返回的对象被认为是只读的,因此对它们进行修改可能会在不同线程中的调用者产生冲突。

测试

只有session结束之后,数据才会从一级缓存中存放到二级缓存中去。

5 第三方缓存EhCache

导入相应依赖

在对应的mapper配置文件中开启EhCache。

<cache type = "org.mybatis.caches.ehcache.EhcacheCache" />

新建一个ehcache.xml文件

今日份分享已结束,请大家多多包涵和指点!

相关推荐

Vue 技术栈(全家桶)(vue technology)

Vue技术栈(全家桶)尚硅谷前端研究院第1章:Vue核心Vue简介官网英文官网:https://vuejs.org/中文官网:https://cn.vuejs.org/...

vue 基础- nextTick 的使用场景(vue的nexttick这个方法有什么用)

前言《vue基础》系列是再次回炉vue记的笔记,除了官网那部分知识点外,还会加入自己的一些理解。(里面会有部分和官网相同的文案,有经验的同学择感兴趣的阅读)在开发时,是不是遇到过这样的场景,响应...

vue3 组件初始化流程(vue组件初始化顺序)

学习完成响应式系统后,咋们来看看vue3组件的初始化流程既然是看vue组件的初始化流程,咋们先来创建基本的代码,跑跑流程(在app.vue中写入以下内容,来跑流程)...

vue3优雅的设置element-plus的table自动滚动到底部

场景我是需要在table最后添加一行数据,然后把滚动条滚动到最后。查网上的解决方案都是读取html结构,暴力的去获取,虽能解决问题,但是不喜欢这种打补丁的解决方案,我想着官方应该有相关的定义,于是就去...

Vue3为什么推荐使用ref而不是reactive

为什么推荐使用ref而不是reactivereactive本身具有很大局限性导致使用过程需要额外注意,如果忽视这些问题将对开发造成不小的麻烦;ref更像是vue2时代optionapi的data的替...

9、echarts 在 vue 中怎么引用?(必会)

首先我们初始化一个vue项目,执行vueinitwebpackechart,接着我们进入初始化的项目下。安装echarts,npminstallecharts-S//或...

无所不能,将 Vue 渲染到嵌入式液晶屏

该文章转载自公众号@前端时刻,https://mp.weixin.qq.com/s/WDHW36zhfNFVFVv4jO2vrA前言...

vue-element-admin 增删改查(五)(vue-element-admin怎么用)

此篇幅比较长,涉及到的小知识点也比较多,一定要耐心看完,记住学东西没有耐心可不行!!!一、添加和修改注:添加和编辑用到了同一个组件,也就是此篇文章你能学会如何封装组件及引用组件;第二能学会async和...

最全的 Vue 面试题+详解答案(vue面试题知识点大全)

前言本文整理了...

基于 vue3.0 桌面端朋友圈/登录验证+60s倒计时

今天给大家分享的是Vue3聊天实例中的朋友圈的实现及登录验证和倒计时操作。先上效果图这个是最新开发的vue3.x网页端聊天项目中的朋友圈模块。用到了ElementPlus...

不来看看这些 VUE 的生命周期钩子函数?| 原力计划

作者|huangfuyk责编|王晓曼出品|CSDN博客VUE的生命周期钩子函数:就是指在一个组件从创建到销毁的过程自动执行的函数,包含组件的变化。可以分为:创建、挂载、更新、销毁四个模块...

Vue3.5正式上线,父传子props用法更丝滑简洁

前言Vue3.5在2024-09-03正式上线,目前在Vue官网显最新版本已经是Vue3.5,其中主要包含了几个小改动,我留意到日常最常用的改动就是props了,肯定是用Vue3的人必用的,所以针对性...

Vue 3 生命周期完整指南(vue生命周期及使用)

Vue2和Vue3中的生命周期钩子的工作方式非常相似,我们仍然可以访问相同的钩子,也希望将它们能用于相同的场景。...

救命!这 10 个 Vue3 技巧藏太深了!性能翻倍 + 摸鱼神器全揭秘

前端打工人集合!是不是经常遇到这些崩溃瞬间:Vue3项目越写越卡,组件通信像走迷宫,复杂逻辑写得脑壳疼?别慌!作为在一线摸爬滚打多年的老前端,今天直接甩出10个超实用的Vue3实战技巧,手把...

怎么在 vue 中使用 form 清除校验状态?

在Vue中使用表单验证时,经常需要清除表单的校验状态。下面我将介绍一些方法来清除表单的校验状态。1.使用this.$refs...

取消回复欢迎 发表评论: