5年码农的我整理了MyBatis从入门到实战,想精通看这一篇就够了
yuyutoo 2025-01-01 23:26 2 浏览 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文件
今日份分享已结束,请大家多多包涵和指点!
相关推荐
- 二十三种设计模式之-模板方法模式
-
这是我写二十三种设计模式第二篇文章。这个系列我将持续写下去,欢迎大家关注,点赞和收藏。模板方法模式1.模板方法模式(TemplateMethodPattern)又叫模板模式,在一个抽象的类中,公开...
- 从 Java 代码逆向工程生成 UML 类图和序列图
-
前言本文面向于那些软件架构师,设计师和开发人员,他们想使用IBM?Rational?SoftwareArchitect从Java?源代码来逆向工程生成UML类和序列图。逆向工程经常...
- 作为程序员,还在手动画流程图、类图?看看这个神器
-
老板看不懂你写的代码,要求你补充流程图。。。客户看不懂你的代码,要求画流程图。。。新同事看不懂你的代码,要求画流程图。。。此时此刻,你的内心是崩溃的。。。曾几何时,我也和你一样崩溃。。。...
- 使用 seaborn 绘制 12 类图
-
你好,我是zhenguo今晚分享一个很不错的seaborn可视化实战入门材料,这个实战教程来自于kaggle,使用的是美国警察开枪数据集,大小1M,一共5个csv文件使用seaborn作...
- 分享一个从源码快速生成UML类图的插件——PlantUML Parser
-
前言相信每一位程序员都分析过源码,在分析源码过程中,除了了解代码实现的功能(业务逻辑),还需要深入下去了解程序代码的执行过程以及结构,往往在了解代码执行过程(动态模型)前,先对代码的结构(静态模型)有...
- 需求分析-类图建模
-
...
- 还能这么玩?用VsCode画类图、流程图、时序图、...不要太爽
-
软件设计中,有好几种图需要画,比如流程图、类图、组件图等,我知道大部分人画流程图一般都会用微软的viso绘制,我之前也是这个习惯。viso画图有个不好的地方是需要时刻去调整线条和边框已达到简洁美观,今...
- UML:类图关系总结
-
UML类图几种关系的总结,泛化=实现>组合>聚合>关联>依赖在UML类图中,常见的有以下几种关系:泛化(Generalization),实现(Reali...
- 小白进阶之路:一文读懂UML-类图
-
UML类图(UnifiedModelingLanguageClassDiagram)是一种用于可视化和描述系统中类、属性、方法以及它们之间关系的图形化表示方法。我在大学时,学习这个知识总是容易...
- 餐饮系统大拆解:用类图拆解员工结构与工作职责(1)
-
编辑导语:利用类图这一方式,产品经理可以更清晰地梳理设计思路,进而推动后续方案的迭代优化,同时结合类图梳理,团队内也能降低沟通成本。具体应该如何拆解?本篇文章里,作者结合餐饮系统,对类图拆解和梳理做了...
- 软件开发设计文档之「类图」
-
对象是系统中用来描述客观事物的一个实体,它由对象标识(名称)、属性(状态、数据、成员变量)和服务(操作、行为、方法)三个要素组成,它们被封装为一个整体,以接口的形式对外提供服务。而类则是对具有相同属性...
- 如何绘制「UML类图」?附内容详解和优质实例分析!
-
下面这篇文章是笔者整理分析的关于如何绘制「UML类图」的相关内容,大家一起来看看吧!UML图有很多种,但是并非必须掌握所有的UML图,才能完整系统分析和设计工作。一般说来,在UML图中,只要掌握类图、...
- UML统一建模语言系列二:类图设计方法及最佳实践
-
一、前言...
- 类图(Class Diagram)
-
类图(ClassDiagram):类(Class)封装了数据和行为,是面向对象的重要组成部分,它是具有相同属性、操作、关系的对象集合的总称。类一般由三部分组成:类名(Class):每个类都必须有一个...
- 类图怎么画?简单快速绘制类图的软件
-
类图是显示模型中的类、类的内部结构和其他类的关系的图表,用来描述系统的结构化设计。类图是由类、包等元素和内容相互连接组成,是最常用的UML图。类图是描述系统中的类以及它们之间的关系的图表,它的主要作用...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)