MyBatis学习二:Mapper代理开发、配置文件完成增删改查、注解开发
前言
公司要求没办法,前端也要了解一下后端知识,这里记录一下自己的学习
学习教程:黑马mybatis教程全套视频教程,2天Mybatis框架从入门到精通
文档:
https://mybatis.net.cn/index.html
Mapper代理开发
目的
- 解决原生方式中的硬编码
- 简化后期执行sql
Mapper代理要求
- 定义与sql映射文件同名的Mapper接口,并且将Mapper接口和sql映射文件放置在同一目录下
- 设置sql映射文件的
namespace
属性未Mapper接口全限定名 - 在Mapper接口中定义方法,方法名就算sql映射文件中sql语句的id,并保持参数类型和返回值类型一致
- 编码
- 通过
SqlSession
的getMapper
方法获取Mapper接口的代理对象 - 调用对应方法完成sql的执行
- 通过
备注: 如果Mapper接口名称和sql映射文件名称相同,并且在同一目录下,则可以使用包扫描的方式简化sql映射文件的加载。
定义Mapper接口,并且将Mapper接口和sql映射文件放置在同一目录下
1、在java/com/mapper
下创建一个UserMapper
接口文件
2、在resources
下创建相同的目录结构,com/example/mapper
,将UserMapper.xml
移入到该目录下
3、注意,创建完成后在文件管理器里看一下目录结构是否正确,com.example.mapper
是三个文件夹,不是一个文件夹。视频里也说了,这个很重要
设置sql映射文件的namespace
属性未Mapper接口全限定名
修改UserMapper.xml
文件里的namespace
属性,修改为com.example.mapper.UserMapper
在Mapper接口中定义方法,方法名就算sql映射文件中sql语句的id,并保持参数类型和返回值类型一致
更新mybatis-config.xml里映射文件的地址
修改测试类
// 3、执行sql语句,查询所有数据
// 指定要执行的sql语句,这里传入对应的标识,对应UserMapper.xml文件中<select id="selectAllUser" resultType="User">
// List<User> userList = sqlSession.selectList("test.selectAllUser");
// 3.1 获取UserMapper接口的代理对象
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
// 3.2 执行查询
List<User> userList = userMapper.selectAllUser();
执行结果
使用包扫描的方式简化sql映射文件的加载
mybatis-config.xml
配置文件完成增删改查
基本步骤
- 编写Mapper接口方法
- 编写sql语句,在sql映射文件里
实体类属性与表字段不一致
例如:类属性是userName
,表里的属性是user_name
。这样导致无法正确的查询出结果。
解决方式是设置:resultMap
原代码
<select id="selectAllUser" resultType="com.example.pojo.User">select * from user;
</select>
修改后
<resultMap id="userResultMap" type="com.example.pojo.User"><!-- property 属性是指对应的 Java 类的属性,column 属性是指对应的数据库表的字段名 --><!-- 主键映射--><id property="id" column="id"/><!--普通列映射--><result property="userName" column="user_name"/>
</resultMap><select id="selectAllUser" resultMap="userResultMap">select * from user;
</select>
根据id进行数据查询
UserMapper
接口
// 根据id查询
User selectUserById(Integer id);
UserMapper.xml
<select id="selectUserById" parameterType="int" resultType="com.example.pojo.User">select * from user where id = #{id};
</select>
测试
// 根据id进行查询
User user = userMapper.selectUserById(1);
System.out.println("用户:" + user);
补充:
parameterType
用于指定参数类型,这个也可以省略,因为mapper接口李的函数已经指定了参数类型- 参数使用
#{参数}
进行传递,除了#{}
还存在${}
。#{}
会将参数替换为?
,可以防止sql注入;${}
会显示为实际的值,会存在sql注入问题。 - 特殊字符,因为是在
xml
中写sql,所以<
和<=
会与xml
标签冲突,<
可以使用<
代替,<=
使用<=
代替
条件查询
多条件查询
- 参数同属于一个对象时
UserMapper
接口里的函数
List<User> selectUserById(User user);
测试的代码
// 根据id和年龄进行查询
User userParam = new User();
userParam.setId(2);
userParam.setAge(30);
List<User> user = userMapper.selectUserById(userParam);
System.out.println("用户:" + user.size());
UserMapper.xml
里的sql
<select id="selectUserById" resultType="com.example.pojo.User">select * from user where id <#{id} and age < #{age};
</select>
使用对象的这种方式要注意:当你调用这个方法时,将会把 user.getId()
的值传递给SQL语句中的#{id}
参数。但是要确保 User
类中有一个名为 id
的属性,并且提供了对应的getter
方法。
- 参数不属于一个对象时
UserMapper
接口里的函数,需要使用@Param
指定参数名称
List<User> selectUserById(@Param("id") int id, @Param("age") int age);
测试的代码
List<User> user = userMapper.selectUserById(3,10);
UserMapper.xml
里的sql
,#{}
里的变量要与@Param
定义的保持一致
<select id="selectUserById" resultType="com.example.pojo.User">select * from user where id <#{id} and age < #{age};
</select>
- 参数是一个
map
对象
这里只需要注意map
对象的key
要与sql
里的参数保持一致
动态条件查询
if
条件判断
<select id="selectUserById" resultType="com.example.pojo.User">select * from user where id <#{id}<if test="age !=-1 and age <100">and age < #{age}</if>
</select>
如果id
和 age
都需要判断时可以采用下面的方式
<select id="selectUserById" resultType="com.example.pojo.User">select * from user<where><if test="id!=-1">and id = #{id}</if><if test="age!=-1">and age = #{age}</if></where></select>
但条件动态查询
使用choose,when,otherwise
,类似于switch,case,default
select * from user where
<choose><when test="id!=-1">id = #{id}</when><when test="age!=-1">age = #{age}</when><otherwise>1=1</otherwise>
</choose>
或者
select * from user
<where><choose><when test="id!=-1">id = #{id}</when><when test="age!=-1">age = #{age}</when></choose>
</where>
添加、修改
添加
UserMapper
接口
// 添加用户void addUser(User user);
UserMapper.xml
<!--values对应的是类里的属性 -->
<!-- 设置useGeneratedKeys和keyProperty后可以在新增成功后返回主键值-->
<insert id="addUser" useGeneratedKeys="true" keyProperty="id">insert into user(name, age, email)values (#{name}, #{age}, #{email});
</insert>
测试代码
User newUser = new User();
newUser.setName("李四");
newUser.setAge(18);
newUser.setEmail("123@qq.com");
userMapper.addUser(newUser);
// 这里要手动提交一下事务
sqlSession.commit();
System.out.println("新添加的id是:" + newUser.getId());
动态修改
<update id="updateUser">update user<set><if test="name!=null">name = #{name},</if><if test="age!=null">age = #{age},</if><if test="email!=null">email = #{email}</if></set><where>id = #{id}</where></update>
删除
删除一个
<!-- 单条数据的删除-->
<delete id="deleteUserById">delete from userwhere id = #{id}
</delete>
批量删除
void deleteUserByIds(@Param("ids") int[] ids);
<delete id="deleteUserByIds">delete from userwhere id in<foreach collection="ids" item="id" open="(" separator="," close=")">#{id}</foreach>
</delete>
参数传递
当接口的参数是Collection
、List
、Array
、多个参数时需要使用@Param
注解,来给参数命名,确保sql
中可以正确识别,如上面的批量删除。
注解开发
前面开发时sql
语句都是写在xml
,配置文件里。这里使用注解可以代替配置文件的方式,更加方便。
当然如果sql
语句比较复杂还是使用配置文件的方式。
注解
- 查询:
@Select
- 添加:
@insert
- 修改:
@Update
- 删除:
Delete
以查询用户为例
xml
配置方式
List<User> selectAllUser();
<!-- 查询所有用户-->
<select id="selectAllUser" resultMap="userResultMap">select *from user;
</select>
注解方式
@Select("select * from user where name = #{name}")
User selectUserByName(String name);
相关文章:

MyBatis学习二:Mapper代理开发、配置文件完成增删改查、注解开发
前言 公司要求没办法,前端也要了解一下后端知识,这里记录一下自己的学习 学习教程:黑马mybatis教程全套视频教程,2天Mybatis框架从入门到精通 文档: https://mybatis.net.cn/index.html Mapper代理开发 目的 解决…...

【React系列】受控非受控组件
本文来自#React系列教程:https://mp.weixin.qq.com/mp/appmsgalbum?__bizMzg5MDAzNzkwNA&actiongetalbum&album_id1566025152667107329) 一. refs 的使用 在React的开发模式中,通常情况下不需要、也不建议直接操作DOM原生,但是某些…...
OpenCV-Python(22):2D直方图
目标 了解图像的2D直方图绘制2D直方图 介绍 在前面的部分我们介绍了如何绘制一维直方图,之所以称为一维,是因为我们只考虑了图像的一个特征:灰度值。但是在2D 直方图中我们就需要考虑两个图像特征。对于彩色图像的直方图通常情况下我们需要…...
Kubernetes 100个常用命令
本文简单总结关于使用 Kubectl 进行 Kubernetes 诊断的指南。列出了 100 个 Kubectl 命令,这些命令对于诊断 Kubernetes 集群中的问题非常有用。这些问题包括但不限于: 集群信息 Pod 诊断 服务诊断 部署诊断 网络诊断 持久卷和持久卷声明诊断 资源…...

labuladong日常刷题-差分数组 | LeetCode 1109航班预定统计 | 花式遍历 151反转字符串里的单词
差分数组–前缀和数组的升级 LeetCode 1109 航班预定统计 2024.1.1 题目链接labuladong讲解[链接] class Solution { public:vector<int> corpFlightBookings(vector<vector<int>>& bookings, int n) {//构建航班人数数组,数组大小为n,初…...

HbuilderX中的git的使用
原文链接https://blog.csdn.net/Aom_yt/article/details/119924356...
LeetCode每日一题 | 1944. 队列中可以看到的人数
文章目录 队列中可以看到的人数题目描述问题分析程序代码(Golang 版本) 队列中可以看到的人数 题目描述 原题链接 有 n 个人排成一个队列,从左到右 编号为 0 到 n - 1 。给你以一个整数数组 heights ,每个整数 互不相同ÿ…...
React16源码: JSX2JS及React.createElement源码实现
JSX 到 Javascript 的转换 React中的 JSX 类似于 Vue中的template模板文件,Vue是基于编译时将template模板转换成render函数在React中,JSX是类似于html和javascript混编的语法,而javascript是真的javascript, html并非真的html它的可阅读性可…...
整理composer安装版本的python脚本
整理composer安装版本的python脚本 脚本实现的功能是去除composer安装命令后的版本号 def remove_version_numbers(commands):"""Remove version numbers from composer require commands.Args:commands (list of str): List of composer require commands.Retu…...

十、基本对话框大集合(Qt5 GUI系列)
目录 一、设计需求 二、实现代码 三、代码解析 四、总结 一、设计需求 Qt提供了很多标准的对话框。例如标准文件对话框(QFileDialog)、标准颜色对话框(QColorDialog)、标准字体对话框 (QFontDialog)、标准输入对话框 (QInputDialog) 及消息对话框 (QMessageBox)。本文展示各…...
大A又跌了
才开盘几天,又开始下跌了。生活更加苦难。期待高深算法。...

This error originates from a subprocess, and is likely not a problem with pip
我遇这个问题是的原因是包名错误 注意检查包名...

数据库基础知识1
关系模型的程序员不需熟悉数据库的存取路径 在3层模式结构中,___I___是数据库的核心和关键,___Ⅱ___通常是模式的子集,数据库模式的描述提供给用户,____Ⅲ__的描述存储在硬盘上。Ⅰ.模式Ⅱ. 外模式Ⅲ. 内模式 数据库中,数据的物理独立性是指用户的应用程序与存储在磁盘上数据库…...

【GO语言卵细胞级别教程】01.GO基础知识
01.GO基础知识 目录 01.GO基础知识1.GO语言的发展历程2.发展历程3.Windowns安装4.VSCode配置5.基础语法5.1 第一段代码5.2 GO执行的流程5.3 语法规则5.4 代码风格5.5 学习网址 1.GO语言的发展历程 Go语言是谷歌公司于2007年开始开发的一种编程语言,由Robert Griese…...
215.【2023年华为OD机试真题(C卷)】按身高和体重排排队(排序题-JavaPythonC++JS实现)
🚀点击这里可直接跳转到本专栏,可查阅顶置最新的华为OD机试宝典~ 本专栏所有题目均包含优质解题思路,高质量解题代码(Java&Python&C++&JS分别实现),详细代码讲解,助你深入学习,深度掌握! 文章目录 一. 题目-按身高和体重排排队二.解题思路三.题解代码Pyt…...
虚函数(C++)
四、多态4.1 虚函数 四、多态 多态性是面向对象程序设计语言的又一重要特征,多态(polymorphism)通俗的讲,就是用一个相同的名字定义许多不同的函数,这些函数可以针对不同数据类型实现相同或类似的功能,即所…...
力扣25题: K 个一组翻转链表
【题目链接】力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台,解题代码如下: class Solution {public ListNode reverseKGroup(ListNode head, int k) {ListNode curNode head;ListNode groupHead, groupTail head, lastGrou…...

网络安全应急响应工具之-流量安全取证NetworkMiner
在前面的一些文章中,用了很多的章节介绍流量分析和捕获工具wireshark。Wireshark是一款通用的网络协议分析工具,非常强大,关于wireshark的更多介绍,请关注专栏,wireshark从入门到精通。本文将介绍一个专注于网络流量取…...
http 401 错误
HTTP 401 错误表示未被授权,指客户端通过请求头中的身份验证数据进行身份验证,服务器返回401状态码表示身份验证失败。HTTP 401 错误通常与身份验证和授权相关的 Web 请求有关。 一、HTTP 401错误的定义 HTTP 401 错误是 HTTP 状态码的一种。由于服务器…...

Docker-Compose部署Redis(v7.2)哨兵模式
文章目录 一、前提准备1. 主从集群2. 文件夹结构 二、配置文件1. redis server配置文件2. redis sentinel配置文件3. docker compose文件 三、运行四、测试 环境 docker desktop for windows 4.23.0redis 7.2 一、前提准备 1. 主从集群 首先需要有一个redis主从集群&#x…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...

8k长序列建模,蛋白质语言模型Prot42仅利用目标蛋白序列即可生成高亲和力结合剂
蛋白质结合剂(如抗体、抑制肽)在疾病诊断、成像分析及靶向药物递送等关键场景中发挥着不可替代的作用。传统上,高特异性蛋白质结合剂的开发高度依赖噬菌体展示、定向进化等实验技术,但这类方法普遍面临资源消耗巨大、研发周期冗长…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的一体化测试平台,覆盖应用全生命周期测试需求,主要提供五大核心能力: 测试类型检测目标关键指标功能体验基…...

令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南 在数字化营销时代,邮件列表效度、用户参与度和网站性能等指标往往决定着创业公司的增长成败。今天,我们将深入解析邮件打开率、网站可用性、页面参与时…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...
Python ROS2【机器人中间件框架】 简介
销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...

嵌入式学习笔记DAY33(网络编程——TCP)
一、网络架构 C/S (client/server 客户端/服务器):由客户端和服务器端两个部分组成。客户端通常是用户使用的应用程序,负责提供用户界面和交互逻辑 ,接收用户输入,向服务器发送请求,并展示服务…...
【LeetCode】3309. 连接二进制表示可形成的最大数值(递归|回溯|位运算)
LeetCode 3309. 连接二进制表示可形成的最大数值(中等) 题目描述解题思路Java代码 题目描述 题目链接:LeetCode 3309. 连接二进制表示可形成的最大数值(中等) 给你一个长度为 3 的整数数组 nums。 现以某种顺序 连接…...