MyBatis系统学习(五)——MyBatis的注解开发

news2024/9/23 4:15:11

一、MyBatis 注解开发概述

MyBatis 是一个支持自定义 SQL、存储过程和高级映射的持久层框架。除了传统的 XML 方式外,MyBatis 还提供了注解的方式进行 SQL 操作定义。这种注解方式将 SQL 直接写在 Java 接口方法上,简化了开发过程,便于维护和使用。

常见的 MyBatis 注解有:

  • @Select:执行查询 SQL。
  • @Insert:执行插入 SQL。
  • @Update:执行更新 SQL。
  • @Delete:执行删除 SQL。
  • @Results / @Result:定义结果集的映射。
  • @One / @Many:用于处理关联查询(一对一和一对多)。
  • @Param:为参数命名,方便在 SQL 中使用。
  • @Options:配置 SQL 执行选项(如返回自增主键)。

二、基于注解的单表增删改查操作

1. 查询操作 (@Select)

@Select 注解用于执行查询操作,可以直接将查询结果映射为 POJO(Plain Old Java Object)对象或列表。

示例:

public interface UserMapper {
    // 查询所有用户
    @Select("SELECT * FROM user")
    List<User> getAllUsers();

    // 根据 ID 查询用户
    @Select("SELECT * FROM user WHERE id = #{id}")
    User getUserById(int id);
}

在此示例中:

  • #{id} 是一个占位符,表示方法参数 id,MyBatis 会自动将参数值传入 SQL 中。
2. 插入操作 (@Insert)

@Insert 用于插入数据,传递的对象属性将自动映射到 SQL 中的字段。

示例:

public interface UserMapper {
    // 插入新用户
    @Insert("INSERT INTO user(name, age) VALUES(#{name}, #{age})")
    int insertUser(User user);
}
3. 更新操作 (@Update)

@Update 用于更新表中的数据,可以根据特定条件(如 ID)更新记录。

示例:

public interface UserMapper {
    // 根据 ID 更新用户信息
    @Update("UPDATE user SET name=#{name}, age=#{age} WHERE id=#{id}")
    int updateUser(User user);
}
4. 删除操作 (@Delete)

@Delete 用于删除数据,通常根据 ID 或其他条件来删除记录。

示例:

public interface UserMapper {
    // 根据 ID 删除用户
    @Delete("DELETE FROM user WHERE id = #{id}")
    int deleteUser(int id);
}

三、基于注解的关联查询

1. 一对一关联查询 (@One)

@One 注解用于处理一对一的关联查询。假设有两张表:user(用户表)和 user_detail(用户详情表),它们通过 user.iduser_detail.user_id 相关联。

示例:

public interface UserMapper {
    // 查询用户及其详情信息
    @Select("SELECT * FROM user WHERE id = #{id}")
    @Results({
        @Result(column = "id", property = "id"),
        @Result(column = "name", property = "name"),
        @Result(column = "age", property = "age"),
        @Result(column = "id", property = "userDetail",
                one = @One(select = "com.example.mapper.UserDetailMapper.getUserDetailByUserId"))
    })
    User getUserWithDetail(int id);
}

public interface UserDetailMapper {
    @Select("SELECT * FROM user_detail WHERE user_id = #{userId}")
    UserDetail getUserDetailByUserId(int userId);
}

在这个例子中,UserMapper 通过 @One 注解来处理 user_detail 的关联查询,UserDetailMapper 用来查询用户的详情信息。

2. 一对多关联查询 (@Many)

@Many 注解用于处理一对多的关联查询。假设用户表 user 和订单表 order 之间是一个用户对应多个订单的关系。

示例:

public interface UserMapper {
    // 查询用户及其订单信息
    @Select("SELECT * FROM user WHERE id = #{id}")
    @Results({
        @Result(column = "id", property = "id"),
        @Result(column = "name", property = "name"),
        @Result(column = "age", property = "age"),
        @Result(column = "id", property = "orders",
                many = @Many(select = "com.example.mapper.OrderMapper.getOrdersByUserId"))
    })
    User getUserWithOrders(int id);
}

public interface OrderMapper {
    @Select("SELECT * FROM `order` WHERE user_id = #{userId}")
    List<Order> getOrdersByUserId(int userId);
}

在此示例中,@Many 用于关联查询用户的订单。OrderMapper 中的 getOrdersByUserId 方法根据用户 ID 查询所有订单。

四、多对多关联查询

多对多关系通常需要第三张关联表。例如,student 表与 course 表之间存在多对多关系,学生可以选修多门课程,课程也可以被多个学生选修。中间表 student_course 用来记录学生和课程的对应关系。

数据库结构:
CREATE TABLE student (
    id INT PRIMARY KEY AUTO_INCREMENT,
    name VARCHAR(50)
);

CREATE TABLE course (
    id INT PRIMARY KEY AUTO_INCREMENT,
    name VARCHAR(50)
);

CREATE TABLE student_course (
    student_id INT,
    course_id INT,
    FOREIGN KEY (student_id) REFERENCES student(id),
    FOREIGN KEY (course_id) REFERENCES course(id)
);
实现多对多查询:

1. StudentMapper:

public interface StudentMapper {
    // 查询学生及其所选课程
    @Select("SELECT * FROM student WHERE id = #{id}")
    @Results({
        @Result(column = "id", property = "id"),
        @Result(column = "name", property = "name"),
        @Result(column = "id", property = "courses",
                many = @Many(select = "com.example.mapper.CourseMapper.getCoursesByStudentId"))
    })
    Student getStudentWithCourses(int id);
}

2. CourseMapper:

public interface CourseMapper {
    // 根据学生 ID 查询所选课程
    @Select("SELECT course.* FROM course " +
            "JOIN student_course ON course.id = student_course.course_id " +
            "WHERE student_course.student_id = #{studentId}")
    List<Course> getCoursesByStudentId(int studentId);
}

在这个例子中,StudentMapperCourseMapper 协同工作,实现了多对多的查询。通过 student_course 中间表将学生和课程的关系关联起来。

五、其它注解的详细说明

1. @Param 注解

@Param 注解用于为方法中的参数命名,特别是当方法有多个参数时,使用 @Param 可以显式指定参数名,从而避免混淆,提升代码的可读性。

示例:

public interface UserMapper {
    // 根据用户名和年龄查询用户
    @Select("SELECT * FROM user WHERE name = #{name} AND age = #{age}")
    User getUserByNameAndAge(@Param("name") String name, @Param("age") int age);
}

在此示例中,@Param("name")@Param("age") 指定了 SQL 中的参数名称,确保在 SQL 语句中正确映射。

2. @Options 注解

@Options 注解用于配置 SQL 执行时的一些选项。例如,如果表的主键是自增的,可以使用 @Options 来配置插入操作返回生成的主键。

示例:

public interface UserMapper {
    // 插入用户并返回自增主键
    @Insert("INSERT INTO user(name, age) VALUES(#{name}, #{age})")
    @Options(useGeneratedKeys = true, keyProperty = "id")
    int insertUser(User user);
}

useGeneratedKeys = true 表示自动生成的主键会被返回并填充到 user 对象的 id 属性中。

六、综合案例

假设我们有一个包含用户、订单、产品的系统。用户可以有多个订单,每个订单包含多个产品(多对多关系)。让我们通过 MyBatis 注解实现该场景。

1. 数据库表结构
CREATE TABLE user (
    id INT PRIMARY KEY AUTO_INCREMENT,
    name VARCHAR(50)
);

CREATE TABLE product (
    id INT PRIMARY KEY AUTO

_INCREMENT,
    name VARCHAR(100)
);

CREATE TABLE `order` (
    id INT PRIMARY KEY AUTO_INCREMENT,
    user_id INT,
    FOREIGN KEY (user_id) REFERENCES user(id)
);

CREATE TABLE order_product (
    order_id INT,
    product_id INT,
    FOREIGN KEY (order_id) REFERENCES `order`(id),
    FOREIGN KEY (product_id) REFERENCES product(id)
);
2. 实体类
public class User {
    private int id;
    private String name;
    private List<Order> orders;  // 用户的订单
}

public class Order {
    private int id;
    private int userId;
    private List<Product> products;  // 订单包含的产品
}

public class Product {
    private int id;
    private String name;
}
3. Mapper 接口

UserMapper:

public interface UserMapper {
    // 查询用户及其订单及产品
    @Select("SELECT * FROM user WHERE id = #{id}")
    @Results({
        @Result(column = "id", property = "id"),
        @Result(column = "name", property = "name"),
        @Result(column = "id", property = "orders",
                many = @Many(select = "com.example.mapper.OrderMapper.getOrdersByUserId"))
    })
    User getUserWithOrdersAndProducts(int id);
}

OrderMapper:

public interface OrderMapper {
    // 根据用户 ID 查询订单
    @Select("SELECT * FROM `order` WHERE user_id = #{userId}")
    @Results({
        @Result(column = "id", property = "id"),
        @Result(column = "user_id", property = "userId"),
        @Result(column = "id", property = "products",
                many = @Many(select = "com.example.mapper.ProductMapper.getProductsByOrderId"))
    })
    List<Order> getOrdersByUserId(int userId);
}

ProductMapper:

public interface ProductMapper {
    // 根据订单 ID 查询产品
    @Select("SELECT product.* FROM product " +
            "JOIN order_product ON product.id = order_product.product_id " +
            "WHERE order_product.order_id = #{orderId}")
    List<Product> getProductsByOrderId(int orderId);
}
4. 综合查询操作

调用 getUserWithOrdersAndProducts 方法,可以获取用户、该用户的所有订单以及每个订单包含的产品。

七、总结

通过 MyBatis 注解开发,开发者可以灵活地定义 SQL 操作,简化 XML 配置文件的使用。单表增删改查、一对一、一对多、多对多等关联查询都可以通过注解方式实现,注解方式具有更好的可读性和维护性。此外,合理使用 @Param@Options 注解,能够提高 SQL 操作的灵活性和易用性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2156507.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

02_RabbitMQ消息丢失解决方案及死信队列

一、数据丢失 第一种&#xff1a;生产者弄丢了数据。生产者将数据发送到 RabbitMQ 的时候&#xff0c;可能数据就在半路给搞丢了&#xff0c;因为网络问题&#xff0c;都有可能。 第二种&#xff1a;RabbitMQ 弄丢了数据。MQ还没有持久化自己挂了。 第三种&#xff1a;消费端…

Attention is All You Need精读

原文开头&#xff0c;不是搬运 dog。All attention is all you need Abstract 摘要 这篇文章的内容源于基于编码器-解码器架构的RNN模型&#xff0c;在其他人的工作中&#xff0c;我们可以发现注意力机制对于提升编码器-解码器架构模型的性能很重要。这篇文章提出了一个比较简…

《代码整洁之道:程序员的职业素养》

作者&#xff1a;【美】Robert C. Martin 第1章 专业主义 专业主义就意味着担当责任&#xff0c;软件开发太复杂了&#xff0c;不可能没什么 bug。但很不幸&#xff0c;这并不能为你开脱。人体太复杂了&#xff0c;不可能尽知其全部&#xff0c;但医生仍要发誓不伤害病人。如…

隐藏excel单元格数据的两个方法

在Excel中&#xff0c;公式是用来计算数据和结果的非常重要的一部分。但是&#xff0c;有时候您可能希望隐藏公式&#xff0c;以保护其不被他人修改或查看。那么今天小编就来给大家分享隐藏excel单元格数据的方法。 一、使用“隐藏”功能 在Excel中&#xff0c;我们还可以使用…

ZYNQ学习--AXI总线协议

一、AXI 总线简介 AXI&#xff08;Advanced Extensible Interface&#xff09;高级拓展总线是AMBA&#xff08;Advanced Microcontroller Bus Architecture&#xff09;高级微控制总线架构中的一个高性能总线协议&#xff0c;由ARM公司开发。AXI总线协议被广泛应用于高带宽、低…

大语言模型超参数调优:开启 AI 潜能的钥匙

前言 在人工智能的广袤领域中&#xff0c;大语言模型&#xff08;LLM&#xff09;凭借其强大的实力&#xff0c;不断重塑着我们对机器理解语言的认知。然而&#xff0c;要使这些模型在特定应用场景中发挥最大效能&#xff0c;关键在于巧妙调整其超参数。本文将引领你深入探究 …

x-cmd pkg | bat: cat 命令现代化替代品,终端用户必备工具

目录 简介快速上手安装使用与第三方工具组合使用 功能特点竞品和相关作品进一步阅读 简介 bat 是由 github.com/sharkdp 用 Rust 开发的 cat 命令现代化替代品。它比 cat 命令扩展了更多的现代化功能&#xff0c;如语法高亮、自动分页、Git集成等&#xff0c;能为用户提供更为…

python如何跨文件调用自己定义的函数

当自己定义函数过多时&#xff0c;只有一个python文件时代码会很长&#xff0c;不易理清代码框架&#xff0c;比如下面这段代码&#xff0c;如何隐藏具体函数细节呢&#xff1f;也就是把def函数放到另外一个python文件里步骤如下&#xff1a; 一个python文件代码篇幅过长 imp…

结构体对齐、函数传参、库移植

结构体字节对齐 按固定位大小匹配地址&#xff0c;a:10b:1020位 <32位4字节 202040位>32位 所以ab20作为一个int型&#xff0c;int c:20 单独做4个字节&#xff08;int&#xff09; 101020 &#xff08;int&#xff09;4个字节 &#xff0c;20&#xff08;int&#x…

算法之逻辑斯蒂回归(Logistic regression)

简介&#xff1a;个人学习分享&#xff0c;如有错误&#xff0c;欢迎批评指正。 逻辑斯蒂回归&#xff08;Logistic Regression&#xff09;是统计学中一种广泛应用于二分类问题的算法。它的主要目标是预测二分类问题中的事件发生的概率。尽管名字里有“回归”&#xff0c;但逻…

wordpress迁移到别的服务器

wordpress论坛网站搭建 于2023/11/16写的该文章 一-配置环境 配置LNMP&#xff08;linuxnginxmysqlphpphpmyadmin&#xff09;环境或者LAMP&#xff08;apache&#xff09; 可以选择集成了这些软件的套件 下载链接&#xff1a;https://www.xp.cn/download.html 手动下载这…

https加密原理

以为http的数据都是以明文传送&#xff0c;会有很大的安全问题&#xff0c;所以出现的https协议。https就是在http协议的基础上增加了一个安全层&#xff0c;可以对数据进行加密和解密(例如SSL、TLS等)。 https加密解密的原理&#xff1a;证书非对称加密对称加密 在讲解原理前…

Python爬虫之urllib模块详解

Python爬虫入门 此专栏为Python爬虫入门到进阶学习。 话不多说&#xff0c;直接开始吧。 urllib模块 Python中自带的一个基于爬虫的模块&#xff0c;其实这个模块都几乎没什么人用了&#xff0c;我就随便写写了。 - 作用&#xff1a;可以使用代码模拟浏览器发起请求。&…

2024 年最新前端ES-Module模块化、webpack打包工具详细教程(更新中)

模块化概述 什么是模块&#xff1f;模块是一个封装了特定功能的代码块&#xff0c;可以独立开发、测试和维护。模块通过导出&#xff08;export&#xff09;和导入&#xff08;import&#xff09;与其他模块通信&#xff0c;保持内部细节的封装。 前端 JavaScript 模块化是指…

【Pytorch】一文快速教你高效使用torch.no_grad()

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 博主简介 博主致力于嵌入式、Python、人工智能、C/C领域和各种前沿技术的优质博客分享&#xff0c;用最优质的内容带来最舒适的…

2024年中国研究生数学建模竞赛A/C/D/E题全析全解

问题一&#xff1a; 针对问题一&#xff0c;可以采用以下低复杂度模型&#xff0c;来计算风机主轴及塔架的疲劳损伤累积程度。 建模思路&#xff1a; 累积疲劳损伤计算&#xff1a; 根据Palmgren-Miner线性累积损伤理论&#xff0c;元件的疲劳损伤可以累积。因此&#xff0c;…

基于SpringBoot+Vue的商城积分系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 精品专栏&#xff1a;Java精选实战项目源码、Python精…

手机在网状态查询接口如何用C#进行调用?

一、什么是手机在网状态查询接口&#xff1f; 手机在网状态查询接口是利用实时数据来对手机号码在运营商网络中的状态进行查询的工具&#xff0c;包括正常使用状态、停机状态、不在网状态、预销户状态等。 二、手机在网状态查询适用哪些场景&#xff1f; 例如&#xff1a;商…

【线程池】ThreadPoolExecutor应用

ThreadPoolExecutor应用 每一步的坚持与积累,都是铸就高薪和大牛的必经的修炼 哈哈,不吹牛逼了,今天来分享最近在提升中的学习总结,无论是对在职场还是求职,看完,我相信都会有些许的收获和成长 也难得过了一个悠闲点的周末,哈哈哈,一起奥利给!! 本文总纲: 1.为什么要自定义线程…

联合体的用法和用联合体判断大小端存储

像结构体⼀样&#xff0c;联合体也是由⼀个或者多个成员构成&#xff0c;这些成员可以不同的类型。但是编译器只为最⼤的成员分配⾜够的内存空间。联合体的特点是所有成员共⽤同⼀块内存空间。所 以联合体也叫&#xff1a;共⽤体。 给联合体其中⼀个成员赋值&#xff0c;其他成…