• MyBatis映射关系


    目录

    数据库的配置

    一.映射关系一对一

    1.映射关系 1 对 1-基本介绍

    2.映射关系 1 对 1-映射方式

    3.应用实例

    3.1方式一

    方式二:

    重点解析:

    注解的方式实现

    注意事项和细节

    二.映射关系多对一

    1.基本介绍

    2.注意细节

    3.映射方式

    4.代码实现

    4.1数据库表

    4.2UserMapper.xml文件

    4.3PetMapper.xml文件

    注解实现多对 1 映射


    数据库的配置

    1. CREATE TABLE person
    2. (
    3. id INT PRIMARY KEY AUTO_INCREMENT,
    4. NAME VARCHAR(32) NOT NULL DEFAULT '',
    5. card_id INT ,
    6. FOREIGN KEY (card_id) REFERENCES idencard(id)
    7. )CHARSET utf8;
    1. -- 创建 mybatis_idencard 表
    2. CREATE TABLE idencard
    3. (
    4. id INT PRIMARY KEY AUTO_INCREMENT,
    5. card_sn VARCHAR(32) NOT NULL DEFAULT '' )CHARSET utf8 ;
    6. INSERT INTO idencard VALUES(1,'111111111111110');
    7. INSERT INTO person VALUES(1,'张三',1);
    8. SELECT * FROM person;
    9. SELECT * FROM idencard

    一.映射关系一对一

    1.映射关系 1 1-基本介绍

    基本介绍
    1. 项目中 1 1 的关系是一个基本的映射关系,比如 :Person( ) --- IDCard( 身份证 )
    2. 我们看看再 MyBatis 中如何实现 1 1 的处理 .
    注意细节
    1 1 , 我们这里就研究一下单向 1 1 即可

    2.映射关系 1 1-映射方式

    映射方式
    1. 通过配置 XxxMapper.xml 实现 1 1 [ 配置方式 ]
    2. 通过注解的方式实现 1 1 [ 注解方式 ]

    3.应用实例

    3.1方式一

    通过配置 XxxMapper.xml 的方式来实现下面的 1 1 的映射关系,实现
    联查询 , 通过 person 可以获取到对应的 idencard 信息

     然后建立各自的javabean--->两个表

     然后我们就可以开始重头戏了开始建立IdenCardMapper.java和IdenCardMapper.xml 

    1. public interface IdenCardMapper {
    2. //根据id获取到身份证序列号
    3. public IdenCard getIdenCardById(Integer id);
    1. //通过Person的id获取到Person,包括这个Person关联的IdenCard对象[级联查询]
    2. public Person getPersonById(Integer id);
    1. <mapper namespace="com.hong.mapper.PersonMapper">
    2. <resultMap id="PersonResultMap" type="Person">
    3. <id property="id" column="id"/>
    4. <result property="name" column="name"/>
    5. <association property="card" javaType="IdenCard">
    6. <result property="id" column="id"/>
    7. <result property="card_sn" column="card_sn"/>
    8. association>
    9. resultMap>
    10. <select id="getPersonById" parameterType="Integer"
    11. resultMap="PersonResultMap">
    12. SELECT * FROM `person`,`idencard` WHERE `person`.id = #{id}
    13. AND `person`.card_id = `idencard`.id
    14. select>
    15. mapper>

    方式二:

    通过配置 XxxMapper.xml 的方式来实现下面的 1 1 的映射关系,实现级
    联查询 , 通过 person 可以获取到对应的 identcard 信息

    1. 修改 PersonMapper.java PersonMapper.xml 使用第 2 种映射方式 , 完成 1 1 映射
    关系

    1. //通过Person的id获取到Person,包括这个Person关联的IdenCard对象,方式2
    2. public Person getPersonById2(Integer id);
    3. //编写方法,通过card_id 查询得到person对象/数据
    4. public Person getPersonByCardId(Integer cardId);
    1. <resultMap id="PersonResultMap2" type="Person">
    2. <id property="id" column="id"/>
    3. <result property="name" column="name"/>
    4. <association property="card" column="card_id"
    5. select="com.hong.mapper.IdenCardMapper.getIdenCardById" />
    6. resultMap>
    7. <select id="getPersonById2" parameterType="Integer" resultMap="PersonResultMap2">
    8. SELECT * FROM `person` WHERE `id` = #{id}
    9. select>
    10. mapper>
    1. mybatis第二种方式核心思想: 将这个多表联查,分解成单表操作 , 这样简洁,而且易于维护 ,推荐
    2. 而且可以复用你已经写好的方法 -组合
    3. property="card": 表示 Person对象的 card 属性
    4. column="card_id" 这个是
    SELECT * FROM `person` WHERE `id` = #{id}  返回的 字段 card_id 信息/数据
    5. 返回的 字段 card_id 信息/数据 作为getIdenCardById入参, 来执行

    重点解析

    先执行下面的select语句,然后将执行的结果中的card_id传参给getIdenCardById,这个时候我们就直接返回的就是确定这个值的IdenCard的值了。

    好,现在我们试着来说明一下啥叫将多表级联变成现在的单表操作。

    就是我们先来思考一下MySQL中的多表查询,是不是有一个子查询,你要是知道这个就差不多理解了一半了。我们数据库中的子查询是先将一个查询结果返回给另外一个SQL语句进行操作。那么一对一映射其实本质也是一样,我们先将一个结果作为参数然后传入到一个方法中,这个方法就可以运用这个结果去执行他的操作。

    注解的方式实现

    通过注解的方式来实现下面的 1 1 的映射关系,实现级联查询 , 通过 person 可以获取到
    对应的 identcard 信息
    在实际开发中还是 推荐使用配置方式
    1. public interface PersonMapperAnnotation {
    2. //这里注解实现方法
    3. //说明: 注解的形式就是对前面xml配置方式的体现
    4. //这里同学们可以结合前面讲解的xml配置时,加入的注释来理解
    5. @Select("SELECT * FROM `person` WHERE `id` = #{id}")
    6. @Results({
    7. @Result(id = true, property = "id", column = "id"),
    8. @Result(property = "name", column = "name"),
    9. @Result(property = "card", column = "card_id",
    10. one = @One(select = "com.hong.mapper.IdenCardMapper.getIdenCardById"))
    11. })
    12. public Person getPersonById(Integer id);
    13. }

    注解的方式也就是xml形式的另外一种体现

    1. public class PersonMapperAnnotationTest {
    2. //属性
    3. private SqlSession sqlSession;
    4. private PersonMapperAnnotation personMapperAnnotation;
    5. //初始化
    6. @Before
    7. public void init() {
    8. //获取到sqlSession
    9. sqlSession = MyBatisUtils.getSqlSession();
    10. personMapperAnnotation = sqlSession.getMapper(PersonMapperAnnotation.class);
    11. }
    12. @Test
    13. public void getPersonById() {
    14. Person person = personMapperAnnotation.getPersonById(1);
    15. System.out.println("person----" + person);
    16. if(sqlSession != null) {
    17. sqlSession.close();
    18. }
    19. }
    20. }

    注意事项和细节

    1. 表是否设置外键 , MyBatis 进行对象 / 级联映射没有影响
    2. 举例 : 去掉 person 表的外键 , 进行测试 , 依然可以获取相应的级联对象

    二.映射关系多对一

    1.基本介绍

    1. 项目中多对 1 的关系是一个基本的映射关系 , 多对 1, 也可以理解成是 1 对多 .
    2. User --- Pet : 一个用户可以养多只宠物
    3. Dep ---Emp : 一个部门可以有多个员工

    2.注意细节

    1. 我们直接讲 双向的多对一的关系 ,单向的多对一比双向的多对一简单。
    2. 在实际的项目开发中 , 要求会使用双向的多对一的映射关系
    3. 说明:什么是 双向的多对一 的关系 : 比如通过 User 可以查询到对应的 Pet, 反过
    来,通过 Pet 也可以级联查询到对应的 User 信息 .
    4. 多对多的关系,是在多对 1 的基础上扩展即可

    3.映射方式

    1. 方式 1 :通过配置 XxxMapper.xml 实现多对 1
    2. 方式 2 :通过注解的方式实现 多对 1

    4.代码实现

    4.1数据库表

    1. CREATE TABLE mybatis_user
    2. (id INT PRIMARY KEY AUTO_INCREMENT,
    3. NAME VARCHAR(32) NOT NULL DEFAULT '' )CHARSET=utf8 ;
    4. CREATE TABLE mybatis_pet
    5. (id INT PRIMARY KEY AUTO_INCREMENT,
    6. nickname VARCHAR(32) NOT NULL DEFAULT '',
    7. user_id INT ,
    8. FOREIGN KEY (user_id) REFERENCES mybatis_user(id)
    9. )CHARSET=utf8 ;
    10. INSERT INTO mybatis_user
    11. VALUES(NULL,'宋江'),(NULL,'张飞');
    12. INSERT INTO mybatis_pet
    13. VALUES(1,'黑背',1),(2,'小哈',1);
    14. INSERT INTO mybatis_pet
    15. VALUES(3,'波斯猫',2),(4,'贵妃猫',2);
    16. SELECT * FROM mybatis_user;
    17. SELECT * FROM mybatis_pet;

    4.2UserMapper.xml文件

    1. <mapper namespace="com.hong.mapper.UserMapper">
    2. <resultMap id="UserResultMap" type="User">
    3. <id property="id" column="id"/>
    4. <result property="name" column="name"/>
    5. <collection property="pets" column="id" ofType="Pet"
    6. select="com.hong.mapper.PetMapper.getPetByUserId"/>
    7. resultMap>
    8. <select id="getUserById" parameterType="Integer" resultMap="UserResultMap">
    9. SELECT * FROM `mybatis_user` WHERE `id` = #{id}
    10. select>
    11. mapper>

    4.3PetMapper.xml文件

    1. <mapper namespace="com.hong.mapper.PetMapper">
    2. <resultMap id="PetResultMap" type="Pet">
    3. <id property="id" column="id"/>
    4. <result property="nickname" column="nickname"/>
    5. <association property="user" column="user_id"
    6. select="com.hong.mapper.UserMapper.getUserById" />
    7. resultMap>
    8. <select id="getPetByUserId" parameterType="Integer" resultMap="PetResultMap">
    9. SELECT * FROM `mybatis_pet` WHERE `user_id` = #{userId}
    10. select>
    11. <select id="getPetById"
    12. parameterType="Integer"
    13. resultMap="PetResultMap">
    14. SELECT * FROM `mybatis_pet` WHERE `id` = #{id}
    15. select>
    16. mapper>

    注解实现多对 1 映射

    需求说明 : 通过注解的方式来实现下面的多对 1 的映射关系,实现级联查询 , 完成前面完成
    的任务,通过 User-->Pet 也可 Pet->User , 在实际开发中 推荐使用配置方式来
    1. public interface UserMapperAnnotation {
    2. //通过id获取User对象
    3. /**
    4. * 1. 注解的配置就是对应的Mapper.xml文件配置的,改写
    5. * 2.
    6. * 1、一定要想一想我们前面1-1是如何实现
    7. * 2、配置/实现 public User getUserById(Integer id);
    8. * 3、思路(1) 先通过user-id 查询得到user信息 (2) 再根据user-id查询对应的pet信息
    9. * 并映射到User-List pets
    10. *
    11. *
    12. *
    13. * 1. ofType="Pet" 指定返回的集合中存放的数据类型Pet
    14. * 2. collection 表示 pets 是一个集合
    15. * 3. property="pets" 是返回的user对象的属性 pets
    16. * 4. column="id" SELECT * FROM `mybatis_user` WHERE `id` = #{id} 返回的id字段对应的值
    17. * -->
    18. *
    19. * select="com.hong.mapper.PetMapper.getPetByUserId"/>
    20. *
    21. *
    22. */
    23. @Select("SELECT * FROM `mybatis_user` WHERE `id` = #{id}")
    24. @Results({
    25. @Result(id = true, property = "id", column = "id"),
    26. @Result(property = "name", column = "name"),
    27. //这里请小伙伴注意,pets属性对应的是集合
    28. @Result(property = "pets",
    29. column = "id",
    30. many = @Many(select = "com.hong.mapper.PetMapperAnnotation.getPetByUserId"))
    31. })
    32. public User getUserById(Integer id);
    33. }
    1. public interface PersonMapperAnnotation {
    2. //这里注解实现方法
    3. //说明: 注解的形式就是对前面xml配置方式的体现
    4. //这里同学们可以结合前面老师讲解的xml配置时,加入的注释来理解
    5. @Select("SELECT * FROM `person` WHERE `id` = #{id}")
    6. @Results({
    7. @Result(id = true, property = "id", column = "id"),
    8. @Result(property = "name", column = "name"),
    9. @Result(property = "card", column = "card_id",
    10. one = @One(select = "com.hong.mapper.IdenCardMapper.getIdenCardById"))
    11. })
    12. public Person getPersonById(Integer id);
    13. }
    1. public interface PetMapperAnnotation {
    2. //通过User的id来获取pet对象,可能有多个,因此使用List接收
    3. /**
    4. * 1、通过User的id来获取pet对象,可能有多个,因此使用List接收
    5. * 2、public List getPetByUserId(Integer userId);
    6. * 3. 完成的思路和前面大体相同.
    7. *
    8. *
    9. *
    10. *
    11. * select="com.hong.mapper.UserMapper.getUserById" />
    12. *
    13. *
    14. */
    15. //id = "PetResultMap" 就是给我们的Results[Result Map] 指定一个名字
    16. //,目的是为了后面复用
    17. @Select("SELECT * FROM `mybatis_pet` WHERE `user_id` = #{userId}")
    18. @Results(id = "PetResultMap", value = {
    19. @Result(id = true, property = "id", column = "id"),
    20. @Result(property = "nickname", column = "nickname"),
    21. @Result(property = "user",
    22. column = "user_id",
    23. one = @One(select = "com.hong.mapper.UserMapperAnnotation.getUserById"))
    24. })
    25. public List getPetByUserId(Integer userId);
    26. //通过pet的id获取Pet对象, 同时会查询到pet对象关联的user对象
    27. /**
    28. *
    29. *
    30. * @ResultMap("PetResultMap") 使用/引用我们上面定义的 Results[ResultMap]
    31. */
    32. @Select("SELECT * FROM `mybatis_pet` WHERE `id` = #{id}")
    33. @ResultMap("PetResultMap")
    34. public Pet getPetById(Integer id);
    35. }
    一对多的思路:
    1.首先我们通过id="getUserById"的select查询获取到user-id对应的所有数据
    2.之后也可以通过user-id获取到pet是的信息
    3.之后我们也可以通过User的id来获取pet对象,可能有多个,因此使用List接收

    简而言之:

            就是我们可以通过主人的id查询到宠物的信息,同时也可以通过宠物的信息查询到主人。

    一对一是我先查询一个id,在通过这个id查询到一个信息,而这个信息是可以查询到另外一个表的信息的。而一对多是我通过一个字段查询到这个字段在这个信息里面的所相关的内容(就像一个id可以查询到主人对应的宠物信息),当然是存放在集合之中的。

  • 相关阅读:
    pthread_mutex_unlock
    dubbo和springcloud问题解决——interface not allow null
    Java实现快速排序以及原理
    Flink容错机制
    【Go】Go 语言切片(Slice)
    SV-6002T-P 网络对讲求助终端,立柱式智慧城市网络对讲求助终端,停车场出入口一键求助终端
    封箱vs拆箱
    我叫:选择排序【JAVA】
    数据结构与算法--贪心算法
    pwn入门(1)二进制基础
  • 原文地址:https://blog.csdn.net/weixin_54107527/article/details/127812142