MyBatis 注解开发详解
MyBatis 注解开发详解
MyBatis 支持使用注解来进行数据库操作。注解方式将 SQL 语句直接写在 Java 接口中,通过注解来完成 CRUD(增删改查)操作,省去了使用 XML 配置的繁琐步骤。这种方式适合简单项目或快速原型开发,因为它更直观,可以直接在代码中查看和修改 SQL 语句。
然而,在实际开发中,更推荐使用 XML 配置的方式进行 MyBatis 开发。这样做的主要好处是,当需要修改 SQL 语句时,不需要更改 Java 代码,避免频繁修改源代码带来的风险。此外,XML 配置文件的形式更有利于 SQL 语句的管理和维护,尤其是在项目复杂度较高、SQL 语句较多的情况下。
以下是使用注解方式进行 MyBatis 开发的详细说明。
一、使用注解完成 CRUD 操作
1. 配置文件 SqlMapConfig.xml
在 SqlMapConfig.xml 中注册 Mapper 接口有两种常见方式:
- 第一种方式:直接引入单个 Mapper 接口类。这种方式适用于小型项目或接口数量较少的情况。
- 第二种方式:引入整个包下的所有接口。这种方式更为灵活,可以自动扫描指定包中的所有接口,适用于较大项目。
<mappers><!-- 第一种方式:class 引入单个接口 --><mapper class="com.qcby.dao.UserAnnoDao"/><!-- 第二种方式:引入整个包下的所有接口 --><package name="com.qcby.dao"/>
</mappers>
2. UserDao 接口及注解
在 UserDao 接口中,通过注解实现对数据库的增删改查操作。常用注解包括:
@Select:用于查询操作。@Insert:用于插入数据。@Update:用于更新数据。@Delete:用于删除数据。@Results和@Result:用于映射查询结果到 Java 对象属性。
import com.qcby.entity.User;
import org.apache.ibatis.annotations.*;import java.util.List;public interface UserDao {// 查询所有用户@Select("SELECT * FROM user")@Results(id = "userMap", value = {@Result(property = "id", column = "id"),@Result(property = "username", column = "username"),@Result(property = "birthday", column = "birthday"),@Result(property = "sex", column = "sex"),@Result(property = "address", column = "address")})List<User> findAll();// 根据 ID 查询用户@Select("SELECT * FROM user WHERE id = #{id}")@ResultMap("userMap")User findById(int id);// 插入新用户@Insert("INSERT INTO user(username, birthday, sex, address) VALUES(#{username}, #{birthday}, #{sex}, #{address})")@SelectKey(statement = "SELECT last_insert_id()", keyProperty = "id", before = false, resultType = Integer.class)int insert(User user);// 更新用户@Update("UPDATE user SET username = #{username}, birthday = #{birthday}, sex = #{sex}, address = #{address} WHERE id = #{id}")int update(User user);// 删除用户@Delete("DELETE FROM user WHERE id = #{id}")int delete(int id);// 查询用户数量@Select("SELECT COUNT(*) FROM user")int findCount();// 模糊查询@Select("SELECT * FROM user WHERE username LIKE #{username}")List<User> findByName(String username);
}
3. UserTest 测试方法
为了验证 UserDao 接口的正确性,可以通过单元测试进行测试。在测试类 UserTest 中,主要步骤包括:
- 初始化 MyBatis 配置,创建
SqlSession。 - 通过
SqlSession获取UserDao接口的代理对象。 - 使用代理对象调用接口方法,执行数据库操作。
- 在测试方法执行前后,分别进行资源的初始化和销毁。
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;import java.io.IOException;
import java.io.InputStream;
import java.util.Date;
import java.util.List;public class UserTest {private InputStream in;private SqlSession session;private UserDao mapper;@Beforepublic void init() throws IOException {in = Resources.getResourceAsStream("SqlMapConfig.xml");SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(in);session = factory.openSession();mapper = session.getMapper(UserDao.class);}@Afterpublic void destroy() throws IOException {session.close();in.close();}@Testpublic void findAll() {List<User> users = mapper.findAll();users.forEach(System.out::println);}@Testpublic void findById() {User user = mapper.findById(4);System.out.println(user);}@Testpublic void insert() {User user = new User();user.setUsername("小美");user.setSex("女");user.setBirthday(new Date());user.setAddress("保定");mapper.insert(user);session.commit();}@Testpublic void update() {User user = new User();user.setId(22);user.setUsername("小美");user.setSex("女");user.setBirthday(new Date());user.setAddress("上海");mapper.update(user);session.commit();}@Testpublic void delete() {mapper.delete(22);session.commit();}@Testpublic void findCount() {int count = mapper.findCount();System.out.println(count);}@Testpublic void findByName() {List<User> users = mapper.findByName("%a%");users.forEach(System.out::println);}
}
二、使用注解完成多对一查询
在多对一的关系中,可以通过注解配置查询结果的映射关系,直接将查询结果映射到关联的 Java 对象中。
1. 多对一立即加载
立即加载的方式在查询主表数据时,会同时加载关联表的数据,适合数据量较小的情况。
StudentDao 接口
@Select("SELECT student.*, teacher.Tname FROM student LEFT JOIN teacher ON student.t_id = teacher.id")
@Results(value = {@Result(property = "id", column = "id"),@Result(property = "Sname", column = "Sname"),@Result(property = "sex", column = "sex"),@Result(property = "age", column = "age"),@Result(property = "t_id", column = "t_id"),@Result(property = "teacher.Tname", column = "Tname")
})
List<Student> getStudent();
2. 多对一延迟加载
延迟加载的方式只有在需要访问关联数据时才会查询关联表的数据,适合数据量较大的情况,可以提高查询性能。
StudentDao 接口
@Select("SELECT * FROM student")
@Results(value = {@Result(property = "id", column = "id"),@Result(property = "Sname", column = "Sname"),@Result(property = "sex", column = "sex"),@Result(property = "age", column = "age"),@Result(property = "teacher", column = "t_id", one = @One(select = "com.qcby.dao.TeacherDao.getTeacher", fetchType = FetchType.LAZY))
})
List<Student> getStudent();
TeacherDao 接口
@Select("SELECT * FROM teacher WHERE id = #{t_id}")
Teacher getTeacher(Integer id);
三、使用注解完成一对多查询
在一对多的关系中,可以通过注解配置关联表的数据映射,直接将关联的数据映射到集合属性中。
TeacherDao 接口
@Select("SELECT * FROM Teacher")
@Results(value = {@Result(property = "id", column = "id"),@Result(property = "Tname", column = "Tname"),@Result(property = "students", column = "id", many = @Many(select = "com.qcby.dao.StudentDao.findByUid", fetchType = FetchType.LAZY))
})
List<Teacher> findAllLazy();
StudentDao 接口
@Select("SELECT * FROM student WHERE t_id = #{t_id}")
Student findByUid(int uid);
总结
MyBatis 注解开发的方式通过将 SQL 语句直接嵌入到 Java 代码中,省去了 XML 配置的繁琐步骤,适合快速开发和简单项目。然而,对于复杂项目,建议使用 XML 配置的方式进行 SQL 语句的管理,以提高代码的可维护性和可扩展性。
相关文章:
MyBatis 注解开发详解
MyBatis 注解开发详解 MyBatis 支持使用注解来进行数据库操作。注解方式将 SQL 语句直接写在 Java 接口中,通过注解来完成 CRUD(增删改查)操作,省去了使用 XML 配置的繁琐步骤。这种方式适合简单项目或快速原型开发,因…...
Kivy App开发之UX控件VideoPlayer视频播放
kivy使用VideoPlayer控件实现视频播放,可以控制视频的播放,暂停,音量调节等功能。 在使用VideoPlayer视频播放器时,可以参考下表属性来设置其样式和触发事件。 属性说明source视频路径,默认为空state视频状态,值play,pause,stop,默认为stopthumbnail显示视频的缩略图…...
简单排序算法
异或运算及异或运算实现的swap方法 ^(异或): ^运算是计算机中的位运算,运算规则为相同为0,不同为1(也被称为无进位相加)。位运算处理效率比常规运算符效率更高。 异或运算遵循的法则: 0^N N N^N 0 异或运算…...
C语言初阶牛客网刷题——JZ17 打印从1到最大的n位数【难度:入门】
1.题目描述 牛客网OJ题链接 题目描述: 输入数字 n,按顺序打印出从 1 到最大的 n 位十进制数。比如输入 3,则打印出 1、2、3 一直到最大的 3 位数 999。 用返回一个整数列表来代替打印n 为正整数,0 < n < 5 示例1 输入&…...
基于springboot+vue的校园二手物品交易系统的设计与实现
开发语言:Java框架:springbootJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7(一定要5.7版本)数据库工具:Navicat11开发软件:eclipse/myeclipse/ideaMaven包:…...
开发环境搭建-2:配置 python 运行环境(使用 uv 管理 python 项目)
在 WSL 环境中配置:WSL2 (2.3.26.0) Oracle Linux 8.7 官方镜像 UV 介绍 uv软件官网(github 需要梯子,没错这个软件的官网真就是 github 页面):https://github.com/astral-sh/uv 中文官网(github 需要梯…...
STM32 ST7735 128*160
ST7735 接口和 STM32 SPI 引脚连接 ST7735 引脚功能描述STM32 引脚连接(示例,使用 SPI1)SCLSPI 时钟信号 (SCK)PA0(SPI1_SCK)SDASPI 数据信号 (MOSI)PA1 (SPI1_MOSI)RST复位信号 (Reset)PA2(GPIO 手动控制)DC数据/命令选择 (D/C)PA3 (GPIO 手…...
【面试总结】FFN(前馈神经网络)在Transformer模型中先升维再降维的原因
FFN(前馈神经网络)在Transformer模型中先升维再降维的设计具有多方面的重要原因,以下是对这些原因的总结: 1.目标与动机 高维映射空间:FFN的设计目的是通过一系列线性变换来拟合一个高维的映射空间,而不仅…...
VB读写ini配置文件将运行文件放入任务计划程序设置为开机自启动
本示例使用设备: https://item.taobao.com/item.htm?spma21dvs.23580594.0.0.52de2c1bWmhJZJ&ftt&id562957272162 Public Declare Function GetPrivateProfileString Lib "kernel32" Alias "GetPrivateProfileStringA" (ByVal lpAppl…...
Java基础 (一)
基础概念及运算符、判断、循环 基础概念 关键字 数据类型 分为两种 基本数据类型 标识符 运算符 运算符 算术运算符 隐式转换 小 ------>>> 大 强制转换 字符串 拼接符号 字符 运算 自增自减运算符 ii赋值运算符 赋值运算符 包括 强制转换 关系运算符 逻辑运算符 …...
数据结构——实验六·散列表
嗨~~欢迎来到Tubishu的博客🌸如果你也是一名在校大学生,正在寻找各种编程资源,那么你就来对地方啦🌟 Tubishu是一名计算机本科生,会不定期整理和分享学习中的优质资源,希望能为你的编程之路添砖加瓦⭐&…...
springboot网上书城
摘 要 在Internet高速发展的今天,我们生活的各个领域都涉及到计算机的应用,其中包括网上书城管理系统的网络应用,在国外网上书城管理系统已经是很普遍的方式,不过国内的书城管理系统可能还处于起步阶段。网上书城管理系统具有网上…...
如何在 Pytest 中使用命令行界面和标记运行测试
关注开源优测不迷路 大数据测试过程、策略及挑战 测试框架原理,构建成功的基石 在自动化测试工作之前,你应该知道的10条建议 在自动化测试中,重要的不是工具 在前文你已经初步尝试编写了代码和单元测试,并且想要确保它能正常运行。…...
不建模,无代码,如何构建一个3D虚拟展厅?
在数字化浪潮的推动下,众多企业正积极探索线上3D虚拟展厅这一新型展示平台,旨在以更加生动、直观的方式呈现其产品、环境与综合实力。然而,构建一个既专业又吸引人的3D虚拟展厅并非易事,它不仅需要深厚的技术支持,还需…...
github汉化
本文主要讲述了github如何汉化的方法。 目录 问题描述汉化步骤1.打开github,搜索github-chinese2.打开项目,打开README.md3.下载安装脚本管理器3.1 在README.md中往下滑动,找到浏览器与脚本管理器3.2 选择浏览器对应的脚本管理器3.2.1 点击去…...
Unity Line Renderer Component入门
Overview Line Renderer 组件是 Unity 中用于绘制连续线段的工具。它通过在三维空间中的两个或两个以上的点的数组,并在每个点之间绘制一条直线。可以绘制从简单的直线到复杂的螺旋线等各种图形。 1. 连续性和独立线条 连续性:Line Renderer 绘制的线条…...
数据库的三级模式结构与两级映像
三级模式结构与两级映像 什么是数据库的三级模式结构?1. 模式(Conceptual Schema,概念模式)定义特点作用示例 2. 外模式(External Schema,外部模式)定义特点作用举例 3. 内模式(Inte…...
TCP断开通信前的四次挥手(为啥不是三次?)
1.四次握手的过程 客户端A发送 FIN(终止连接请求) A:我要断开连接了(FIN)。A进入FIN_WAIT_1状态,表示请求断开,等待对方确认。 服务器B回复 ACK(确认断开请求,但还未准备…...
win32汇编环境,按字节、双字等复制字符的操作
;运行效果 ;win32汇编环境,按字节、双字等复制字符的操作 ;这是汇编的优点之一。我们可以按字节、双字、四字、八字节等复制或挨个检查字符。 ;有时候,在接收到的一串信息中,比如访问网站时,返回的字串里,有很多0值存在࿰…...
.net 项目引用与 .NET Framework 项目引用之间的区别和相同
在 .NET 和 .NET Framework 项目中,引用其他库或项目的方式有一些区别和相同之处。以下是详细的对比: 相同点 引用目的: 目的:无论是 .NET 还是 .NET Framework 项目,引用其他库或项目的主要目的是为了使用这些库或项…...
网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...
python打卡day49
知识点回顾: 通道注意力模块复习空间注意力模块CBAM的定义 作业:尝试对今天的模型检查参数数目,并用tensorboard查看训练过程 import torch import torch.nn as nn# 定义通道注意力 class ChannelAttention(nn.Module):def __init__(self,…...
在rocky linux 9.5上在线安装 docker
前面是指南,后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...
C++使用 new 来创建动态数组
问题: 不能使用变量定义数组大小 原因: 这是因为数组在内存中是连续存储的,编译器需要在编译阶段就确定数组的大小,以便正确地分配内存空间。如果允许使用变量来定义数组的大小,那么编译器就无法在编译时确定数组的大…...
C++.OpenGL (14/64)多光源(Multiple Lights)
多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...
【C++特殊工具与技术】优化内存分配(一):C++中的内存分配
目录 一、C 内存的基本概念 1.1 内存的物理与逻辑结构 1.2 C 程序的内存区域划分 二、栈内存分配 2.1 栈内存的特点 2.2 栈内存分配示例 三、堆内存分配 3.1 new和delete操作符 4.2 内存泄漏与悬空指针问题 4.3 new和delete的重载 四、智能指针…...
Linux系统部署KES
1、安装准备 1.版本说明V008R006C009B0014 V008:是version产品的大版本。 R006:是release产品特性版本。 C009:是通用版 B0014:是build开发过程中的构建版本2.硬件要求 #安全版和企业版 内存:1GB 以上 硬盘…...
Sklearn 机器学习 缺失值处理 获取填充失值的统计值
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 使用 Scikit-learn 处理缺失值并提取填充统计信息的完整指南 在机器学习项目中,数据清…...
Visual Studio Code 扩展
Visual Studio Code 扩展 change-case 大小写转换EmmyLua for VSCode 调试插件Bookmarks 书签 change-case 大小写转换 https://marketplace.visualstudio.com/items?itemNamewmaurer.change-case 选中单词后,命令 changeCase.commands 可预览转换效果 EmmyLua…...
