MyBatis实现多表映射、分页显示、逆向工程

news2025/1/12 12:02:27

目录

一、MyBatis实现多表映射

1.1 实体类设计

1.2 一对一关系实现案例

1.3 对多配置实现案例

1.4 设置自动映射与n张表关联映射

二、MyBatis实现分页功能

2.1 mybatis插件工作原理       

2.2 引入插件与插件的使用

三、逆向工程插件

3.1 什么是逆向工程

3.2 MyBatisX插件导入与使用


一、MyBatis实现多表映射

1.1 实体类设计

设计实体类首先需要知道实体类之间的关系,实体类一般关系如下:

一对一:夫妻之间的关系

一对多|多对一:教师与学生之间的关系

多对多:教师与学生之间的关系

一对一案例(订单与用户):

public class Customer {

  private Integer customerId;
  private String customerName;

}

public class Order {

  private Integer orderId;
  private String orderName;
  private Customer customer;// 体现的是对一的关系

}  

一对多案例(用户与订单):

public class Customer {

  private Integer customerId;
  private String customerName;
  private List<Order> orderList;// 体现的是对多的关系
}
public class Order {

  private Integer orderId;
  private String orderName;
  private Customer customer;// 体现的是对一的关系

}
//查询客户和客户对应的订单集合  不要管!

多表结果实体类设计小技巧: 

对一,属性中包含对方对象

对多,属性中包含对方对象集合

只有真实发生多表查询时,才需要设计和修改实体类,否则不提前设计和修改实体类!

无论多少张表联查,实体类设计都是两两考虑!

在查询映射的时候,只需要关注本次查询相关的属性!例如:查询订单和对应的客户,就不要关注客户中的订单集合!


1.2 一对一关系实现案例

使用上述关系,用户与订单关系

在数据库中创建两张表代码如下:

CREATE TABLE `t_customer` (`customer_id` INT NOT NULL AUTO_INCREMENT, `customer_name` CHAR(100), PRIMARY KEY (`customer_id`) );

CREATE TABLE `t_order` ( `order_id` INT NOT NULL AUTO_INCREMENT, `order_name` CHAR(100), `customer_id` INT, PRIMARY KEY (`order_id`) ); 

INSERT INTO `t_customer` (`customer_name`) VALUES ('c01');

INSERT INTO `t_order` (`order_name`, `customer_id`) VALUES ('o1', '1');
INSERT INTO `t_order` (`order_name`, `customer_id`) VALUES ('o2', '1');
INSERT INTO `t_order` (`order_name`, `customer_id`) VALUES ('o3', '1'); 

注意:
实际开发时,一般在开发过程中,不给数据库表设置外键约束。 原因是避免调试不方便。 一般是功能开发完成,再加外键约束检查是否有bug。

实体类中创造上述两个对应的类与接口。

接口如下:

public interface OrderMapper {
//    根据id查询订单与用户的信息
    public Order selectOrderWithCustomer(Integer orderId);
}

 而实现这个功能的sql语句如下:

select * from t_customer c join t_order o
                            on c.customer_id = o.customer_id
                            where order_id = 1
# 其中order_id结果为方法中查询的参数id

可以看到结果如下:

下一步需要实现结果集对应,代码如下:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<!--此处的namespace修改为对应的接口全类名-->
<mapper namespace="com.alphamilk.mapper.OrderMapper">
<!--     接口中业务方法public Order selectOrderWithCustomer(Integer orderId);-->
    <resultMap id="orderInfoById" type="com.alphamilk.pojo.Order">
<!--        设置主键-->
        <id column="order_id" property="orderId"/>
<!--        设置一般属性与列表对应-->
        <result column="order_name" property="orderName"/>
        <result column="customer_id" property="customerId"/>
<!--        通过association标签进行关联
            其中property填入对象属性名
            javaType填入对象类型
-->
        <association property="customer" javaType="com.alphamilk.pojo.Customer">
<!--            customer的主键-->
            <id column="customer_id" property="customerId"/>
            <result column="customer_name" property="customerName"/>
        </association>
    </resultMap>
    <select id="selectOrderWithCustomer" resultMap="orderInfoById">
#         填入对应sql语句
        select * from t_customer c join t_order o
                                on c.customer_id = o.customer_id
        where order_id = #{id}
    </select>
</mapper>

列表结果与实体集之间的对应如下:

注意:做完这些后需要在mybatis配置中将对应的mapper文件导入其中

<mappers>
<!--        此处的mapper标签使用配置里的UnderGraduate.xml的路径-->
<!--        <mapper resource="mappers/UnderGraduateMapper.xml"/>-->
        <mapper resource="mappers/OrderMapper.xml"/>
    </mappers>

进行测试:

public class CRUDTEST {
    private SqlSession session;

    @BeforeEach
    public void init() throws IOException {
        session = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml")).openSession();
    }

    @Test
    public void Test() throws IOException {

        OrderMapper mapper = session.getMapper(OrderMapper.class);
        Order order = mapper.selectOrderWithCustomer(1);
        System.out.println("订单id:"+order.getOrderId());
        System.out.println("订单名称:"+order.getOrderName());
        System.out.println("订单对应客户id"+order.getCustomerId());
        System.out.println("订单对应客户名称"+order.getCustomer().getCustomerName());
    }
    @AfterEach
    public void clear(){
        session.commit();
        session.close();
    }
}

总结(关键字):

关键词

在“对一”关联关系中,我们的配置比较多,但是关键词就只有:association和**javaType


1.3 对多配置实现案例

在案例中,客户与订单之间的关系是对多的,即一个客户有多个订单信息。现在业务需求是通过客户id查询所有客户信息与订单信息。

客户实体类设计

@Data
public class Customer {

  private Integer customerId;
  private String customerName;
  
//  对多关系,只需要创建对应的链表即可,泛型使用对应类
  private List<Order> orderList;

}

 业务方法对应的sql语言

select * from  t_customer c join  t_order o
                            where  c.customer_id = 1;

业务对应的接口

public interface CustomerMapper {
//业务:根据客户id查询其订单
    public List<Customer> CUSTOMERAndOrder(Integer id);
}

接口对应的映射实现

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<!--此处的namespace修改为对应的接口全类名-->
<mapper namespace="com.alphamilk.mapper.CustomerMapper">

<!--    自定义结果集-->
<resultMap id="customerAndOrder" type="com.alphamilk.pojo.Customer">
    <id column="customer_id" property="customerId"/>
    <result column="customer_name" property="customerName"/>

<!--    通过collection实现集合赋值-->
    <collection property="orderList" ofType="com.alphamilk.pojo.Order">
        <id column="order_id" property="orderId"/>
        <result column="order_name" property="orderName"/>
        <result column="customer_id" property="customerId"/>
<!--        注意:里面的customer不需要赋值-->
    </collection>
</resultMap>

<!--     public Customer CUSTOMERAndOrder(Integer id);-->
    <select id="CUSTOMERAndOrder" resultMap="customerAndOrder">
        select * from  studb.t_customer c join  studb.t_order o
        where  c.customer_id = #{id};
    </select>
</mapper>

功能测试:

@Test
    public void Test() throws IOException {

        CustomerMapper mapper = session.getMapper(CustomerMapper.class);
        List<Customer> customers = mapper.CUSTOMERAndOrder(1);

        for (Customer customer :customers){
            List<Order> orderList = customer.getOrderList();
            System.out.println("用户"+customer.getCustomerName()+"订单信息如下");
            for (Order order : orderList){
                System.out.println(order);
            }
        }
    }


1.4 设置自动映射与n张表关联映射

 在经过sql语句结果集的对应过程中可以发现,结果集的对应是一个十分重复且繁琐的工作(表单的属性十分多的情况下),那么有没有能够解放双手简化代码的方法呢?答案是肯定的。

在Mybatis-config设置中进行设置

    <settings>
<!--        设置自动映射,有没有嵌套都会自动帮我们进行映射result标签的属性与列-->
        <setting name="autoMappingBehavior" value="FULL"/>
    </settings>

 下面介绍如何使用与注意事项

在设置好自动映射后,只需要设置好对应的主键即可正常运行。

上面两个案例,在开启自动映射后代码如下

对一映射关键代码如下:

<!--    自定义结果集-->
<!--     接口中业务方法public Order selectOrderWithCustomer(Integer orderId);-->
    <resultMap id="orderInfoById" type="com.alphamilk.pojo.Order">
<!--        设置order主键-->
        <id column="order_id" property="orderId"/>
        <association property="customer" javaType="com.alphamilk.pojo.Customer">
<!--            设置customer的主键-->
            <id column="customer_id" property="customerId"/>
        </association>
    </resultMap>

 注意:
直接使用自动映射有一个前提要求,那就是需要类中的属性名要与列表中的列名称一一对应才行。否则无法自动映射成功。如果在设计实体类和数据库名称不对应的情况下,可以通过起别名的方式进行自动映射。

数据库列名与类属性名不一致情况下的结果:


二、MyBatis实现分页功能

2.1 mybatis插件工作原理       

问题引出:

        由于前端页面的限制,如果返回的数据量十分庞大,前端可能无法进行数据有效展示,这时候就需要用到分页功能。

想要实现分页功能需要 了解mybatis与插件的工作机制。

程序员在写完sql语句后交给mybatis处理sql语句。而mybatis处理过程中可以引入插件,插件的作用就是可以在写过的sql语句上进行一些特殊的修改。分页插件就是对原本的sql语句进行修改。

想要实现sql语句实现分页功能,只需要使用关键字limit x , y( 其中x为偏移量,y为查询数量)如下案例:

SELECT * FROM your_table
LIMIT 20, 10

其中:x , y遵循以下等式(page 为页数,pageSize为页容量)

x = (page-1)*pageSize

y = pageSize

 所以引入的插件,就是动态得帮助我们输入的sql语句上加上limit 对应的x , y.

但是请注意:

1.不需要自行添加limit结尾。

2.写的sql语句代码不能用“ ;” 结尾,因为插件会在分号后面加上limit关键字导致sql报错


2.2 引入插件与插件的使用

这里使用插件名称为pageHelper。

1.首先需要导入对应的依赖,依赖如下:

<!--        实现mybatis分页插件-->
        <dependency>
            <groupId>com.github.pagehelper</groupId>
            <artifactId>pagehelper</artifactId>
            <version>5.1.11</version>
        </dependency>

 2.在Mybatis-config.xml里面进行配置插件

  <plugins>
        <plugin interceptor="com.github.pagehelper.PageInterceptor">
<!--                                               声明拦截的语言是mysql-->
            <property name="helperDialect" value="mysql"/>
        </plugin>
  </plugins>

 3.插件使用

业务接口与实现的映射都按照原来即可,有区别的在于测试时候,如下案例代码

查询所有大学生信息:

//     TODO:注意使用分页插件的sql语句不能以 分号(;)结尾!!!
//    查询所有数据
    @Select("select * from undergraduate.undergraduate")
    public List<UnderGraduate> selectAll();

使用插件测试代码:

    @Test
    public void Test4() {
        UnderGraduateMapper mapper = session.getMapper(UnderGraduateMapper.class);

//        TODO:注意,如果想要进行两条查询语句进行分页,那么不能放在同一个分页区。需要重新启动!
        // 开启分页插件,第一次查询总条数
        PageHelper.startPage(1, Integer.MAX_VALUE);
        List<UnderGraduate> totalList = mapper.selectAll();
        PageInfo<UnderGraduate> totalPageInfo = new PageInfo<>(totalList);

        // 第二次查询,获取想要的分页信息,一般由前端提供,这里参数表明当前第1页,一页容量为2个大学生成员
        PageHelper.startPage(1, 2);
        List<UnderGraduate> list = mapper.selectAll();
        PageInfo<UnderGraduate> pageInfo = new PageInfo<>(list);

        // 输出信息
        System.out.println("list = " + list);
        System.out.println("total = " + totalPageInfo.getTotal());
        System.out.println("pages = " + pageInfo.getPages());
        System.out.println("hasNextPage = " + pageInfo.isHasNextPage());
        System.out.println("hasPreviousPage = " + pageInfo.isHasPreviousPage());
    }

注意:

需要在查询语句开始之前就开启pageHelper插件进行拦截即这段代码

PageHelper.startPage(1, Integer.MAX_VALUE);


三、逆向工程插件

3.1 什么是逆向工程

        所谓逆向工程,举一个通俗的例子,根据一个人的影子来画出一个人大致的轮廓。这就是所谓的逆向。在Java工程中,由于数据库每一张表都要对于一个Java的实体类。而构造实体类又需要花费大量的时间,所以就有了逆向工程,根据一个数据库的表,反向创建对应的实体表。

        在MyBatis中有一些插件就是实现数据库表单逆向创建实体类的功能,比如MyBatisX插件。在介绍MyBatisX插件的使用前,先介绍MyBatisX插件的工作原理。

  1. 数据库反向生成 XML 配置文件

MyBatisX 会通过 JDBC 连接到数据库,然后自动读取数据库中的表、字段、约束等信息,并按照指定的模板格式生成对应的XML配置文件。

  1. 解析 XML 配置文件并生成 Java 类

MyBatisX 会读取之前生成的XML配置文件,解析其中的信息,并使用 Velocity 等相关技术将其转换成Java类、Mapper接口、XML映射文件等相关代码。

  1. 将生成的代码写入磁盘并导入到项目中

MyBatisX 会将生成的 Java 代码、XML映射文件等写入到指定目录下,同时更新项目的配置文件,以便自动扫描并引入生成的代码。

总的来说,MyBatisX 的工作原理是利用反射机制和代码模板生成技术,通过数据库和 XML 配置文件的交互,实现自动生成 MyBatis 相关的映射文件、POJO 类等代码的功能。这样,可以避免手动编写繁琐的代码,提高开发效率和代码质量。


3.2 MyBatisX插件导入与使用

1.找到设置

2.找到插件,输入MybatisX,安装即可

安装好了,那么如何使用呢?

1.首先需要idea连接上mysql数据源,在idea右侧数据库中进行连接,如果已经连接上数据库则可以跳过。

2.如果没有连接上数据库,则在栏中添加MySQL数据库

填写对应用户与密码后测试连接,若提示连接成功即可

3.连接好后选择对应架构之后,在架构对应的表下右键即可看到插件

 选择图中勾选选项,最后进行finish即可创建对应的实体类。


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

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

相关文章

vue3路由配置

npm下载安装路由 npm install vue-router4 --save新建router目录 新建routes路由表可以分离&#xff0c;也可以写在路由文件下的index.js 想用方式引入&#xff0c;需要新建vite.config.ts文件配置

【Redis】使用Java操作Redis

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是Java方文山&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的专栏《Redis》。&#x1f3af;&#x1f3af; &#x1f4…

文本内容转换成语音播放的工具:Speech Mac

Speech Mac版是一款适用于Mac电脑的语音合成工具。它将macOS语音合成器的所有功能整合到一个易于使用的界面中。通过Speech Mac版&#xff0c;用户可以选择40多种声音和语言&#xff0c;方便地将文本转换为语音。用户可以将文本拖放或粘贴到Speech中&#xff0c;并随时更改语音…

JPA编程使用注解@LastModifiedDate,自动更新修改时间字段

一、背景 jpa编程中&#xff0c;我们一般会设计以下五个字段&#xff1a; 在新增记录的时候&#xff0c;无需对创建时间和修改时间进行赋值。 此外&#xff0c;你还得在类上&#xff0c;增加注解EntityListeners(AuditingEntityListener.class)。 EntityListeners(Auditing…

用中文编程工具给澳大利亚客户定制开发的英文版服装进销存软件应用实例

用中文编程工具给澳大利亚客户定制开发的英文版服装进销存软件应用实例 软件从2016年一直用到现在&#xff0c;而且开的分店也是安装的这个软件&#xff0c;上图是定制打印的格式。 该编程工具不但可以连接硬件&#xff0c;而且可以开发大型的软件。 编程系统化课程总目录及明…

Linux下的IMX6ULL——构建bootloader、内核、文件系统(四)

前言&#xff1a; Linux 平台上有许多开源的嵌入式linux系统构建框架(框架的意思就是工 具)&#xff0c;这些框架极大的方便了开发者进行嵌入式系统的定制化构建&#xff0c;目前比较常 见的有OpenWrt, Buildroot, Yocto,等等。其中Buildroot功能强大&#xff0c;使用 简单&…

CY3-N3的荧光特性Cyanine3 azide星戈瑞

CY3-N3是一种荧光染料&#xff0c;其荧光特性通常取决于其分子结构和环境条件。CY3-N3的激发波长通常位于500到550纳米之间。这表示当暴露在具有适当激发光的条件下时&#xff0c;CY3-N3会吸收光并进入激发态。 CY3-N3的发射波长通常位于550到650纳米之间。这意味着在激发后&a…

Mac连接linux的办法(自带终端和iterm2)

1. 使用Mac自带终端Terminal 1.1 点击右上角的聚焦搜索&#xff0c;再输入终端 1.2 查找linux系统的ip地址 在虚拟机里输入如下命令&#xff0c;找到蓝色区域的就是ip地址 ip addr 如果没有显示ip地址&#xff0c;可以重新安装一下虚拟机&#xff0c;之后确保以太网的连接是打…

竹胶板木板片:多功能建筑模板的理想选择

竹胶板木板片作为一种多功能的建筑模板材料&#xff0c;在各类建筑项目中扮演着重要的角色。其防水、耐磨的特性使其成为庭院、阳台等室外空间制模的理想选择。本文将重点介绍竹胶板木板片的特点以及其在建筑模板领域的应用。 竹胶板木板片由优质的竹材制成&#xff0c;经过胶合…

Java中Deque栈对象的增删查(所有方法详解)

1、Deque栈的增删查方法总结 2、方法增删查 栈顶添加&#xff1a;push、offFirst栈尾添加&#xff1a;add、offer、offerLast栈顶删除&#xff1a;remove、pop、poll、pollFirst栈尾删除&#xff1a;pollLast栈顶查看&#xff1a;peek、peekFirst栈尾查看&#xff1a;peekLast…

Miniconda 使用进阶,把它添加到右键菜单中

因为最近有写 python 的需求&#xff0c;且我是用 Miniconda 来管理本地 python 环境的&#xff0c;所以就涉及到需要频繁的进入到环境的命令行中&#xff0c;每次都需要通过官方提供的快捷方式&#xff0c;然后还要一路设置环境和路径&#xff0c;好烦呀。因为之前添加过 Wind…

OceanMind海睿思加入江苏省勘察设计行业协会,数字化能力受勘察设计行业认可

近日&#xff0c;中新赛克海睿思 通过江苏省勘察设计行业协会八届五次常务理事会议审议批准&#xff0c;正式成为江苏省勘察设计行业协会会员单位。 江苏省勘察设计行业协会 江苏省勘察设计行业协会成立于1986年4月&#xff0c;是我国勘察设计行业成立协会较早的省份之一&…

HarmonyOS 数据管理与应用数据持久化(二)

通过键值型数据库实现数据持久化 场景介绍 键值型数据库存储键值对形式的数据&#xff0c;当需要存储的数据没有复杂的关系模型&#xff0c;比如存储商品名称及对应价格、员工工号及今日是否已出勤等&#xff0c;由于数据复杂度低&#xff0c;更容易兼容不同数据库版本和设备…

FreeRTOS_软件定时器

目录 1. 软件定时器简介 2. 定时器服务/Daemon 任务 2.1 定时器服务任务与队列 2.2 定时器相关配置 2.2.1 configUSE_TIMERS 2.2.2 configTIMER_TASK_PRIORITY 2.2.3 configTIMER_QUEUE_LENGTH 2.2.4 configTIMER_TASK_STACK_DEPTH 2.3 单次定时器和周期定时器 2.4 复…

在公共安全场景下,智能监控如何做到保障安全的同时又最大化地提供便利?

智能监控系统应用的场景十分广泛&#xff0c;其中&#xff0c;公共安全场所的需求尤为重要&#xff0c;为保障公共区域的安全&#xff0c;提升人民群众的归属感&#xff0c;增强公共场所的安全性&#xff0c;智慧安防EasyCVR智能视频监控系统做出了极大努力。具体细节如下&…

33 mysql find_in_set 的实现

前言 这里我们主要是来探讨一下 mysql 中 in 的使用, find_in_set 的使用 这两者 在我们实际应用中应该也是 非常常用的了 测试数据表如下 CREATE TABLE tz_test (id int(11) unsigned NOT NULL AUTO_INCREMENT,field1 varchar(16) DEFAULT NULL,field2 varchar(16) DEFAU…

MobPush智能推送:数智化运营加速用户转化

随着移动应用市场竞争日益激烈&#xff0c;如何保持APP用户活跃度实现商业价值成为无数APP运营者的烦恼。是继续加大力度拉新促活&#xff0c;用庞大的用户体量获得平台商业价值&#xff1f;还是加强对存量用户的精细化运营&#xff0c;激发存量用户付费转化实现弯道超车&#…

python爬虫实战——今日头条新闻数据获取

大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 如果有什么疑惑/资料需要的可以点击文章末尾名片领取源码 第三方库: requests >>> pip install requests 第三方模块安装&#xff1a; win R 输入cmd 输入安装命令 pip install 模块名 (如果你觉得安装速度比较慢, 你…

【前端周报11.03】

前端周汇报11.03 那我们接着上一周的继续往下进行推进上周总结本周工作下周内容 那我们接着上一周的继续往下进行推进 上周总结 上一周的话我其实最主要的工作还是进行了一系列的调研主要的话是针对于我们未来要做的小程序的项目的&#xff0c;为未来开发这个小程序做好一系列…

一整套智慧工地源码,劳务实名制、施工安全管理、绿色施工、危大工程设备监测、视频AI识别功能

智慧工地系统充分利用计算机技术、互联网、物联网、云计算、大数据等新一代信息技术&#xff0c;以PC端&#xff0c;移动端&#xff0c;设备端三位一体的管控方式为企业现场工程管理提供了先进的技术手段。让劳务、设备、物料、安全、环境、能源、资料、计划、质量、视频监控等…