MyBatis系列六: 映射关系多对一

动态SQL语句-更复杂的查询业务需求

  • 官方文档
  • 基本介绍
  • 映射方式
  • 配置Mapper.xml的方式-应用实例
  • 注解的方式实现-应用实例
  • 课后练习

在这里插入图片描述

官方文档

文档地址: https://mybatis.org/mybatis-3/zh_CN/sqlmap-xml.html

基本介绍

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

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

映射方式

1.方式1: 通过配置XxxMapper.xml 实现多对1 [配置方式]
2.方式2: 通过注解的方式实现 多对1 [注解方式]
3.我们都实现代码, 应用举例.

配置Mapper.xml的方式-应用实例

●需求说明: 实现级联查询, 通过userid可以查询到用户信息, 并可以查询到关联的pet信息. 反过来, 通过Petid可以查询到Pet的信息, 并且可以级联查询到它的主人User对象信息
在这里插入图片描述

1.创建user表和pet

USE mybatis;

CREATE TABLE `mybatis_user` (
`id` INT PRIMARY KEY AUTO_INCREMENT,
`name` VARCHAR(32) NOT NULL DEFAULT ''
)CHARSET=utf8

CREATE TABLE `mybatis_pet` (
`id` INT PRIMARY KEY AUTO_INCREMENT,
`nickname` VARCHAR(32) NOT NULL DEFAULT '',
`user_id` INT,
FOREIGN KEY(`user_id`) REFERENCES `mybatis_user`(`id`)
)CHARSET=utf8

INSERT INTO `mybatis_user` VALUES (NULL, '宋江'), (NULL, '卢俊义');
INSERT INTO `mybatis_pet` VALUES (NULL, '波斯猫', 1), (NULL, '旺财', 1);
INSERT INTO `mybatis_pet` VALUES (NULL, '斑点猫', 2), (NULL, '花猫', 2);

SELECT * FROM `mybatis_user`;
SELECT * FROM `mybatis_pet`;

2.创建com.zzw.entity.Petcom.zzw.entity.User. 这里toString方法有问题, 后面揭晓

@Getter
@Setter
public class Pet {
    /**
     * CREATE TABLE `mybatis_pet` (
     * `id` INT PRIMARY KEY AUTO_INCREMENT,
     * `nickname` VARCHAR(32) NOT NULL DEFAULT '',
     * `user_id` INT,
     * FOREIGN KEY(`user_id`) REFERENCES `mybatis_user`(`id`)
     * )CHARSET=utf8
     */
    private Integer id;
    private String nickname;
    //一个pet对应一个主人 User对象
    private User user;

    //toString会造成StackOverFlow
    //@Override
    //public String toString() {
    //    return "Pet{" +
    //            "id=" + id +
    //            ", nickname='" + nickname + '\'' +
    //            ", user=" + user +
    //            '}';
    //}
}
@Getter
@Setter
public class User {
    /**
     * CREATE TABLE `mybatis_user` (
     * `id` INT PRIMARY KEY AUTO_INCREMENT,
     * `name` VARCHAR(32) NOT NULL DEFAULT ''
     * )CHARSET=utf8
     */
    private Integer id;
    private String name;
    //因为一个user可以养多个宠物, mybatis 使用集合体现体现这个关系
    private List<Pet> pets;

    //toString会带来麻烦?=>会造成StackOverFlow
    //@Override
    //public String toString() {
    //    return "User{" +
    //            "id=" + id +
    //            ", name='" + name + '\'' +
    //            ", pets=" + pets +
    //            '}';
    //}
}

3.创建UserMapperPetMapper

public interface UserMapper {
    //通过id获取User对象
    public User getUserById(Integer id);
}
public interface PetMapper {
    //通过User的id来获取pet对象, 可能有多个, 因此使用List接收
    public List<Pet> getPetByUserId(Integer userId);

    //通过pet的id获取pet对象, 同时会查询到pet对象关联的user对象
    public Pet getPetById(Integer id);
}

4.创建PetMapper.xmlUserMapper.xml

<mapper namespace="com.zzw.mapper.PetMapper">
    <!--
    1.配置/实现public List<Pet> getPetByUserId(Integer userId);
    2.通过User的id来获取pet对象, 可能有多个, 因此使用List接收
    3.完成的思路和前面大体相同
    -->
    <resultMap id="petResultMap" type="Pet">
        <id property="id" column="id"/>
        <result property="nickname" column="nickname"/>
        <association property="user" column="user_id"
                     select="com.zzw.mapper.UserMapper.getUserById"/><!--先写完, 再去实现getUserById方法-->
    </resultMap>
    <select id="getPetByUserId" parameterType="Integer"
            resultMap="petResultMap">
        SELECT * FROM `mybatis_pet` WHERE `user_id` = #{userId}
    </select>
</mapper>
<mapper namespace="com.zzw.mapper.UserMapper">
    <!--解读
    1.想一想前面的1对1怎么实现的
    2.配置/实现public User getUserById(Integer id);
    3.通过id获取User对象
    4.思路(1) 先通过user-id 查询得到用户信息 (2) 再根据user-id查询对应的pet信息
      并映射到List<Pet> pets
    -->
    <resultMap id="userResultMap" type="User">
        <id property="id" column="id"/>
        <result property="name" column="name"/>
        <!--解读: 因为pets属性是集合, 因此这里需要使用collection标签来处理
        1.ofType="Pet" 指定返回的集合中存放的数据类型Pet
        2.collection 表示 pets 是一个集合
        3.property="pets" 是返回的user对象的属性 pets
        4.column="id" select * from `mybatis_user` where `id` = #{id} 返回的id字段对应的值
        -->
        <collection property="pets" column="id" ofType="Pet"
                    select="com.zzw.mapper.PetMapper.getPetByUserId"/>
    </resultMap>
    <select id="getUserById" parameterType="Integer"
            resultMap="userResultMap">
        select * from `mybatis_user` where `id` = #{id}
    </select>
</mapper>

5.新建UserMapperTest PetMapperTest

public class UserMapperTest {
    //属性
    private SqlSession sqlSession;
    private UserMapper userMapper;

    //初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        userMapper = sqlSession.getMapper(UserMapper.class);
        System.out.println("userMapper--" + userMapper);
    }

    @Test
    public void getUserById() {
        User user = userMapper.getUserById(2);
        System.out.println("user信息--" + user.getId() + "-" + user.getName());

        List<Pet> pets = user.getPets();
        for (Pet pet : pets) {
            System.out.println("养的宠物信息-" + pet.getId() + "-" + pet.getNickname());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }

    }
}
public class PetMapperTest {
    //属性
    private SqlSession sqlSession;
    private PetMapper petMapper;

    //初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        petMapper = sqlSession.getMapper(PetMapper.class);
        System.out.println("petMapper--" + petMapper.getClass());
    }

    @Test
    public void getPetByUserId() {
        List<Pet> pets = petMapper.getPetByUserId(1);
        for (Pet pet : pets) {
            System.out.println("pet信息-" + pet.getId() + "-"  + pet.getNickname());
            User user = pet.getUser();
            System.out.println("user信息-" + user.getName());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}

6.实现getPetById, 体会复用

<!--说明
1. 注意体会resultMap带来好处, 直接复用
2. 配置/实现public Pet getPetById(Integer id);
3. 通过pet的id获取pet对象
-->
<select id="getPetById" parameterType="Integer"
        resultMap="petResultMap">
    SELECT * FROM `mybatis_pet` where `id` = #{id};
</select>
@Test
public void getPetById() {
    Pet pet = petMapper.getPetById(4);
    System.out.println("pet信息-" + pet.getId() + "-" + pet.getNickname());
    User user = pet.getUser();
    System.out.println("主人信息-" + user.getId() + "-" + user.getName());
    if (sqlSession != null) {
        sqlSession.close();
    }
}

注解的方式实现-应用实例

●需求说明: 通过注解的方式来实现下面的多对1的映射关系, 实现级联查询, 通过userid可以查询到用户信息, 并可以查询到关联的pet信息. 反过来, 通过Petid可以查询到Pet的信息, 并且可以级联查询到它的主人User对象信息

说明: 在实际开发中还是 推荐使用配置方式

1.新建com.zzw.mapper.UserMapperAnnotation接口

/**
 * @author 赵志伟
 * @version 1.0
 * UserMapperAnnotation: 使用注解的方式实现多对一
 */
public interface UserMapperAnnotation {
    /**
     * 1.说明: 注解的形式就是前面xml配置方式的体现
     * 2.这里同学们可以结合前面xml实现
     *  <resultMap id="userResultMap" type="User">
     *         <id property="id" column="id"/>
     *         <result property="name" column="name"/>
     *         <collection property="pets" column="id" ofType="Pet"
     *                     select="com.zzw.mapper.PetMapper.getPetByUserId"/>
     *     </resultMap>
     *     <select id="getUserById" parameterType="Integer"
     *             resultMap="userResultMap">
     *         select * from `mybatis_user` where `id` = #{id}
     *     </select>
     */
    @Select("select * from `mybatis_user` where `id` = #{id}")
    @Results({
            @Result(id = true, property = "id", column = "id"),
            @Result(property = "name", column = "name"),
            //这里请小伙伴注意, pets属性对应的是集合
            @Result(property = "pets", column = "id",
                    many = @Many(select = "com.zzw.mapper.PetMapperAnnotation.getPetByUserId"))
    })
    public User getUserById(Integer id);
}

2.新建com.zzw.mapper.PetMapperAnnotation接口

public interface PetMapperAnnotation {
    /**
     * <resultMap id="petResultMap" type="Pet">
     *     <id property="id" column="id"/>
     *     <result property="nickname" column="nickname"/>
     *     <association property="user" column="user_id"
     *                  select="com.zzw.mapper.UserMapper.getUserById"/>
     * </resultMap>
     * <select id="getPetByUserId" parameterType="Integer"
     *         resultMap="petResultMap">
     *     SELECT * FROM `mybatis_pet` WHERE `user_id` = #{userId}
     * </select>
     */
    //id = "petResultMap" 就是给我们的Results[Result Map] 指定一个名字
    //, 目的是为了后面复用
    @Select("SELECT * FROM `mybatis_pet` WHERE `user_id` = #{userId}")
    @Results(id = "petResultMap", value = {
            @Result(id = true, property = "id", column = "id"),
            @Result(property = "nickname", column = "nickname"),
            @Result(property = "user", column="user_id",
                    one = @One(select = "com.zzw.mapper.UserMapperAnnotation.getUserById"))
    })
    public List<Pet> getPetByUserId(Integer userId);

    /**
     * <select id="getPetById" parameterType="Integer"
     *         resultMap="petResultMap">
     *     SELECT * FROM `mybatis_pet` where `id` = #{id}
     * </select>
     *
     * @ResultMap("petResultMap") 使用/引用我们上面定义的 Results[ResultMap]
     */
    @Select("SELECT * FROM `mybatis_pet` where `id` = #{id}")
    @ResultMap("petResultMap")
    public Pet getPetById(Integer id);
}

3.新建 com.zzw.mapper.UserMapperAnnotationTestcom.zzw.mapper.PetMapperAnnotationTest

public class UserMapperAnnotationTest {
    //属性
    private SqlSession sqlSession;
    private UserMapperAnnotation userMapperAnnotation;

    //初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        userMapperAnnotation = sqlSession.getMapper(UserMapperAnnotation.class);
        System.out.println("userMapperAnnotation--" + userMapperAnnotation.getClass());
    }

    @Test
    public void getUserById() {
        User user = userMapperAnnotation.getUserById(2);
        System.out.println("[注解方式]--user信息--" + user.getId() + "-" + user.getName());

        List<Pet> pets = user.getPets();
        for (Pet pet : pets) {
            System.out.println("养的宠物信息-" + pet.getId() + "-" + pet.getNickname());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}
public class PetMapperAnnotationTest {
    //属性
    private SqlSession sqlSession;
    private PetMapperAnnotation petMapperAnnotation;

    //初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        petMapperAnnotation = sqlSession.getMapper(PetMapperAnnotation.class);
        System.out.println("petMapperAnnotation--" + petMapperAnnotation.getClass());
    }

    @Test
    public void getPetByUserId() {
        List<Pet> pets = petMapperAnnotation.getPetByUserId(1);
        for (Pet pet : pets) {
            System.out.println("pet信息-" + pet.getId() + "-"  + pet.getNickname());
            User user = pet.getUser();
            System.out.println("user信息-" + user.getName());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }

    @Test
    public void getPetById() {
        Pet pet = petMapperAnnotation.getPetById(4);
        System.out.println("pet信息-" + pet.getId() + "-" + pet.getNickname());
        User user = pet.getUser();
        System.out.println("主人信息-" + user.getId() + "-" + user.getName());
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}

课后练习

1.自己设计表 dept(部门) 和 emp(雇员), 是1对多的关系
2.通过查询dept, 可以级联查询得到所有emp的信息
3.通过查询emp, 可以级联查询得到对应的dept信息
4.字段小伙伴可以自己设计, 尽量完成.
- dept(id, dep_name)
- emp(id, name, salary, dept_id)

●代码实现
1.创建 dept表 和 emp

USE mybatis;

CREATE TABLE `dept` (
`id` INT PRIMARY KEY AUTO_INCREMENT,
`dep_name` VARCHAR(64) NOT NULL DEFAULT ''
)CHARSET=utf8

CREATE TABLE `emp` (
`id` INT PRIMARY KEY AUTO_INCREMENT,
`name` VARCHAR(64) NOT NULL DEFAULT '',
`salay` DOUBLE NOT NULL DEFAULT 0.0,
`dept_id` INT,
FOREIGN KEY(`dept_id`) REFERENCES `dept`(`id`)
)CHARSET=utf8

INSERT INTO `dept` VALUES (1, '宣传部'), (2, '技术部');
INSERT INTO `emp` VALUES (1, '赵志伟', 6000, 2), (2, '赵培竹', 6000, 1), (3, '阳光', 12000, 2), (4, '月亮', 12000, 1);

SELECT * FROM `dept`;
SELECT * FROM `emp`;

2.创建com.zzw.entity.dept.javacom.zzw.entity.emp.java

@Getter
@Setter
public class Dept {
    private Integer id;
    private String depName;
    //级联查询, 一对多
    private List<Emp> emps;
}
@Getter
@Setter
public class Emp {
    private Integer id;
    private String name;
    private Double salary;
    //级联查询, 一对一
    private Dept dept;
}

3.创建EmpMapper.javaDeptMapper.java

public interface EmpMapper {
    //通过emp的id查询到emp对象, 同时会查询到emp对象关联的dept对象
    public Emp getEmpById(Integer id);

    //通过dept的id获取emp对象, 可能有多个, 因此用List接收
    public List<Emp> getEmpByDeptId(Integer deptId);
}
public interface DeptMapper {
    //通过dept的id查询到dept对象
    public Dept getDeptById(Integer id);
}

4.创建EmpMapper.xmlDeptMapper.xml

<mapper namespace="com.zzw.mapper.EmpMapper">
    <!--
    1.配置/实现public List<Emp> getEmpByDeptId(Integer deptId);
    2.通过dept的id获取emp对象, 可能有多个, 因此用List接收
    -->
    <resultMap id="EmpResultMap" type="Emp">
        <id property="id" column="id"/>
        <result property="name" column="name"/>
        <result property="salary" column="salary"/>
        <association property="dept" column="dept_id"
                     select="com.zzw.mapper.DeptMapper.getDeptById"/>
    </resultMap>
    <select id="getEmpByDeptId" parameterType="Integer" resultMap="EmpResultMap">
        SELECT * FROM `emp` WHERE dept_id = #{deptId}
    </select>
</mapper>
<mapper namespace="com.zzw.mapper.EmpMapper">
    <!--
    1.配置/实现public Dept getDeptById(Integer id);
    2.通过dept的id查询到dept对象
    3.思路(1) 先通过dept-id 查询到部门信息 (2) 再根据dept-id查询对应的emp信息
      并映射到List<Emp> emps
    -->
    <resultMap id="DeptResultMap" type="Dept">
        <id property="id" column="id"/>
        <result property="depName" column="dep_name"/>
        <!--解读: 因为emps属性是集合, 因此这里需要使用collection标签来处理
        1.ofType="Emp" 指定返回的集合中存放的数据类型Emp
        2.collection 表示 emps 是一个集合
        3.property="emps" 是返回的dept对象的属性 emps
        4.column="id" SELECT * FROM `dept` WHERE `id` = #{id} 返回的id字段的值
        -->
        <collection property="emps" column="id" ofType="Emp"
                    select="com.zzw.mapper.EmpMapper.getEmpByDeptId"/>
    </resultMap>
    <select id="getDeptById" parameterType="Integer" resultMap="DeptResultMap">
        SELECT * FROM `dept` WHERE `id` = #{id}
    </select>
</mapper>

5.创建EmpMapperTestDeptMapperTest

public class EmpMapperTest {
    //属性
    private SqlSession sqlSession;
    private EmpMapper empMapper;

    //编写方法完成初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        empMapper = sqlSession.getMapper(EmpMapper.class);
        System.out.println("empMapper=" + empMapper.getClass());
    }

    @Test
    public void getEmpByDeptId() {
        List<Emp> emps = empMapper.getEmpByDeptId(2);
        for (Emp emp : emps) {
            System.out.println("emp信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
            Dept dept = emp.getDept();
            System.out.println("部门信息-" + dept.getId() + "-" + dept.getDepName());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}
public class DeptMapperTest {
    //属性
    private SqlSession sqlSession;
    private DeptMapper deptMapper;

    //编写方法完成初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        deptMapper = sqlSession.getMapper(DeptMapper.class);
        System.out.println("deptMapper=" + deptMapper.getClass());
    }

    @Test
    public void getDeptById() {
        Dept dept = deptMapper.getDeptById(1);
        System.out.println("部门信息-" + dept.getId() + "-" + dept.getDepName());
        List<Emp> emps = dept.getEmps();
        for (Emp emp : emps) {
            System.out.println("员工信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}

6.实现getEmpById, 体会复用

<!--
1.配置/实现public Emp getEmpById(Integer id);
2.通过emp的id查询到emp对象, 同时会查询到emp对象关联的dept对象
-->
<select id="getEmpById" parameterType="Integer" resultMap="EmpResultMap">
    SELECT * FROM `emp` WHERE `id` = #{id}
</select>
@Test
public void getEmpById() {
    Emp emp = empMapper.getEmpById(2);
    System.out.println("emp信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
    Dept dept = emp.getDept();
    System.out.println("部门信息-" + dept.getId() + "-" + dept.getDepName());
    if (sqlSession != null) {
        sqlSession.close();
    }
}

注解方式实现

1.新建EmpMapperAnnotation接口

public interface EmpMapperAnnotation {

   /**
    * 通过emp的id查询到emp对象, 同时会查询到emp对象关联的dept对象
    * <select id="getEmpById" parameterType="Integer" resultMap="EmpResultMap">
    *     SELECT * FROM `emp` WHERE `id` = #{id}
    * </select>
    */
   @Select("SELECT * FROM `emp` WHERE `id` = #{id}")
   @ResultMap("EmpResultMap")
   public Emp getEmpById(Integer id);

   /**
    * 通过dept的id获取emp对象, 可能有多个, 因此用List接收
    * <resultMap id="EmpResultMap" type="Emp">
    *     <id property="id" column="id"/>
    *     <result property="name" column="name"/>
    *     <result property="salary" column="salary"/>
    *     <association property="dept" column="dept_id"
    *                  select="com.zzw.mapper.DeptMapper.getDeptById"/>
    * </resultMap>
    * <select id="getEmpByDeptId" parameterType="Integer" resultMap="EmpResultMap">
    *     SELECT * FROM `emp` WHERE dept_id = #{deptId}
    * </select>
    */
   @Select("SELECT * FROM `emp` WHERE dept_id = #{deptId}")
   @Results(id = "EmpResultMap", value = {
           @Result(id = true, property = "id", column = "id"),
           @Result(property = "name", column = "name"),
           @Result(property = "salary", column = "salary"),
           @Result(property = "dept", column = "dept_id",
                   one = @One(select = "com.zzw.mapper.DeptMapperAnnotation.getDeptById"))
   })
   public List<Emp> getEmpByDeptId(Integer deptId);
}

2.新建DeptMapperAnnotation接口

public interface DeptMapperAnnotation {
    /**通过dept的id查询到dept对象
     * <resultMap id="DeptResultMap" type="Dept">
     *     <id property="id" column="id"/>
     *     <result property="depName" column="dep_name"/>
     *     解读: 因为emps属性是集合, 因此这里需要使用collection标签来处理
     *     1.ofType="Emp" 指定返回的集合中存放的数据类型Emp
     *     2.collection 表示 emps 是一个集合
     *     3.property="emps" 是返回的dept对象的属性 emps
     *     4.column="id" SELECT * FROM `dept` WHERE `id` = #{id} 返回的id字段的值
     *     -->
     *     <collection property="emps" column="id" ofType="Emp"
     *                 select="com.zzw.mapper.EmpMapper.getEmpByDeptId"/>
     * </resultMap>
     * <select id="getDeptById" parameterType="Integer" resultMap="DeptResultMap">
     *     SELECT * FROM `dept` WHERE `id` = #{id}
     * </select>
     */
    @Select("SELECT * FROM `dept` WHERE `id` = #{id}")
    @Results(id = "DeptResultMap", value = {
            @Result(id = true, property = "id", column = "id"),
            @Result(property = "depName", column = "dep_name"),
            @Result(property = "emps", column = "id",
                    many = @Many(select = "com.zzw.mapper.EmpMapperAnnotation.getEmpByDeptId"))
    })
    public Dept getDeptById(Integer id);
}

3.新建 com.zzw.mapper.EmpMapperAnnotationTestcom.zzw.mapper.DeptMapperAnnotationTest

public class EmpMapperAnnotationTest {
    //属性
    private SqlSession sqlSession;
    private EmpMapperAnnotation empMapperAnnotation;

    //编写方法完成初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        empMapperAnnotation = sqlSession.getMapper(EmpMapperAnnotation.class);
        System.out.println("empMapperAnnotation=" + empMapperAnnotation.getClass());
    }

    @Test
    public void getEmpByDeptId() {
        List<Emp> emps = empMapperAnnotation.getEmpByDeptId(2);
        for (Emp emp : emps) {
            System.out.println("[注解方式]emp信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
            Dept dept = emp.getDept();
            System.out.println("部门信息-" + dept.getId() + "-" + dept.getDepName());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }

    @Test
    public void getEmpById() {
        Emp emp = empMapperAnnotation.getEmpById(2);
        System.out.println("[注解方式]emp信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
        Dept dept = emp.getDept();
        System.out.println("[注解方式]dept信息-" + dept.getId() + "-" + dept.getDepName());
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}
public class DeptMapperAnnotationTest {
    //属性
    private SqlSession sqlSession;
    private DeptMapperAnnotation deptMapperAnnotation;

    //编写方法完成初始化
    @Before
    public void init() {
        sqlSession = MyBatisUtils.getSqlSession();
        deptMapperAnnotation = sqlSession.getMapper(DeptMapperAnnotation.class);
        System.out.println("deptMapperAnnotation=" + deptMapperAnnotation.getClass());
    }

    @Test
    public void getDeptById() {
        Dept dept = deptMapperAnnotation.getDeptById(1);
        System.out.println("[注解方式]dept信息-" + dept.getId() + "-" + dept.getDepName());
        List<Emp> emps = dept.getEmps();
        for (Emp emp : emps) {
            System.out.println("[注解方式]emp信息-" + emp.getId() + "-" + emp.getName() + "-" + emp.getSalary());
        }
        if (sqlSession != null) {
            sqlSession.close();
        }
    }
}

接下来我们学习, mybatis缓存…在这里插入图片描述
💐💐💐💐💐💐💐💐给个赞, 点个关注吧, 各位大佬!💐💐💐💐💐💐💐💐

💐💐💐💐💐💐💐💐祝各位2024年大吉大运💐💐💐💐💐💐💐💐💐💐
请添加图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

搜索python包的说明

当我发现bug时&#xff0c;就怀疑是sns包的版本问题了&#xff08;原代码是原作者以前成功运行的代码&#xff09;&#xff0c;于是直接到网上搜&#xff0c;找到对应的说明文档 根据该示例代码进行改写&#xff1a; 达成目的。

Harbor本地仓库搭建003_Harbor常见错误解决_以及各功能使用介绍_镜像推送和拉取---分布式云原生部署架构搭建003

首先我们去登录一下harbor,但是可以看到,用户名密码没有错,但是登录不上去 是因为,我们用了负债均衡,nginx会把,负载均衡进行,随机分配,访问的 是harbora,还是harborb机器. loadbalancer中 解决方案,去loadbalance那个机器中,然后 这里就是25机器,我们登录25机器 然后去配置…

【尚庭公寓SpringBoot + Vue 项目实战】预约看房与租约管理(完结)

【尚庭公寓SpringBoot Vue 项目实战】预约看房与租约管理&#xff08;完结&#xff09; 文章目录 【尚庭公寓SpringBoot Vue 项目实战】预约看房与租约管理&#xff08;完结&#xff09;1、业务说明2、接口开发2.1、预约看房管理2.1.1.保存或更新看房预约2.1.2. 查询个人预约…

首个AI高考全卷评测结果出分,大模型“考生”表现如何?

内容提要 大部分大模型“考生”语文、英语科目表现良好&#xff0c;但在数学方面还有待加强。阅卷老师点评&#xff0c;在语文科目上&#xff0c;对于语言中的一些“潜台词”&#xff0c;大模型尚无法完全理解。在数学科目上&#xff0c;大模型的主观题回答相对凌乱&#xff0…

2005年上半年软件设计师【下午题】试题及答案

文章目录 2005年上半年软件设计师下午题--试题2005年上半年软件设计师下午题--答案2005年上半年软件设计师下午题–试题

力扣每日一题 6/22 字符串/贪心

博客主页&#xff1a;誓则盟约系列专栏&#xff1a;IT竞赛 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ 2663.字典序最小的美丽字符串【困难】 题目&#xff1a; 如果一个字符串满…

NLP大语言模型的缩放定律

一、简述 ​论文《神经语言模型的缩放定律》包含对交叉熵损失的语言模型性能的经验缩放定律的研究&#xff0c;重点关注Transformer架构。 https://arxiv.org/pdf/2001.08361.pdfhttps://arxiv.org/pdf/2001.08361.pdf 实验表明&#xff0c;测试损失与模型大小、数据集…

基于STM8系列单片机驱动74HC595驱动两个3位一体的数码管

1&#xff09;单片机/ARM硬件设计小知识&#xff0c;分享给将要学习或者正在学习单片机/ARM开发的同学。 2&#xff09;内容属于原创&#xff0c;若转载&#xff0c;请说明出处。 3&#xff09;提供相关问题有偿答疑和支持。 为了节省单片机MCU的IO口资源驱动6个数码管&…

STM32单片机USART串口打印和收发数据

文章目录 1. 串口通信 1.1 串口初始化 1.2 库函数 2. 串口打印 2.1 Serial.c 2.2 Serial.h 2.3 main.c 3. 串口收发数据 3.1 Serial.c 3.2 Serial.h 3.3 main.c 1. 串口通信 对于串口通信的详细解析可以看下面这篇文章 STM32单片机USART串口详解-CSDN博客 STM32单片…

基于java+springboot+vue实现的智慧生活商城系统(文末源码+Lw)244

摘 要 计算机网络发展到现在已经好几十年了&#xff0c;在理论上面已经有了很丰富的基础&#xff0c;并且在现实生活中也到处都在使用&#xff0c;可以说&#xff0c;经过几十年的发展&#xff0c;互联网技术已经把地域信息的隔阂给消除了&#xff0c;让整个世界都可以即时通…

数据中心:AI范式下的内存挑战与机遇

在过去的十年里&#xff0c;数据中心和服务器行业经历了前所未有的扩张&#xff0c;这一进程伴随着CPU核心数量、内存带宽(BW)&#xff0c;以及存储容量的显著增长。这种超大规模数据中心的扩张不仅带来了对计算能力的急剧需求&#xff0c;也带来了前所未有的内存功率密度挑战&…

BigDataCloud 反向地理编码

在当今数字化飞速发展的时代&#xff0c;地理信息的精确获取和游戏数据的深入分析成为众多领域的关键需求。2024 年的今天&#xff0c;技术的创新为我们带来了更为出色的 API 服务。BigDataCloud 反向地理编码服务&#xff0c;能够将经纬度迅速而准确地转换为详细位置信息&…

iOS 中,autoreleasepool 的底层实现

在 iOS 中&#xff0c;autoreleasepool 的底层实现基于 Objective-C 运行时&#xff08;runtime&#xff09;和内存管理机制。 图解说明 Objective-C Runtime 和 Autoreleasepool 的创建 在 Objective-C 中&#xff0c;每次进入一个 autoreleasepool 块时&#xff0c;都会创建…

MySQL之复制(十)

复制 改变主库 确定期望的日志位置 如果有备库和新主库的位置不相同&#xff0c;则需要找到该备库最后一条执行的时间在新主库的二进制日志中相应的位置&#xff0c;然后再执行CHANGE MASTER TO.可以通过mysqlbinlog工具来找到备库执行的最后一条查询&#xff0c;然后在主库上…

宇宙星空星辰美景素材哪里找?高清无水印分享

宇宙星空的美丽总能激发人们的无限遐想和灵感&#xff0c;不仅在科学教育领域&#xff0c;更在电影制作和视觉艺术中占有一席之地。为了帮助您找到高质量的宇宙星空视频素材&#xff0c;以下平台将成为您获取令人难忘天体视频素材的首选。 蛙学府 蛙学府作为新媒体创作者的宝库…

FEP容量瓶生产厂商半导体行业耐强酸强碱耐高低温

FEP容量瓶&#xff0c;氟四六容量瓶&#xff0c;特氟龙容量瓶&#xff0c;耐腐蚀耐高温。广泛应用于ICP-MS、ICP-OES等痕量分析以及同位素分析等实验。地质、电子化学品、半导体分析测试、疾控中心、制药厂、环境检测中心等一些机构定容用。 规格参考&#xff1a;10ml、25ml、5…

JupyterLab使用指南(七):JupyterLab使用 LaTeX 生成数学公式

在 JupyterLab 中&#xff0c;可以使用 LaTeX 语法生成复杂的数学公式。JupyterLab 内置对 LaTeX 的支持&#xff0c;使得我们可以方便地在 notebook 中编写和展示数学公式。以下是详细的步骤和示例。 1. 使用 LaTeX 生成数学公式 LaTeX 是一种专门用于排版数学公式的语言。J…

React实现列表列宽可拖拽

1.中文文档上没有&#xff0c;英文文档上有&#xff08;这个老六&#xff01;&#xff01;&#xff09; <Tableborderedcomponents{{header: {cell: ResizableTitle,},}}columns{mergedColumns}dataSource{data} />React - Resizable column

TCP 和 UDP 可以同时绑定相同的端口吗?

在网络编程中&#xff0c;TCP和UDP都可以绑定到同一个端口上进行通信。TCP和UDP是OSI模型中的传输层协议&#xff0c;它们分别使用不同的端口号来区分不同的应用程序或服务。 TCP&#xff08;Transmission Control Protocol&#xff09;提供了面向连接的、可靠的传输服务&…

面向服务的架构(Service-Oriented Architecture, SOA)

目录 前言1. SOA的基本概念1.1 定义和特点1.2 核心原则 2. SOA的优势与挑战2.1 优势2.2 挑战 3. SOA的实现技术3.1 Web服务3.2 微服务架构3.3 企业服务总线&#xff08;ESB&#xff09; 4. SOA在现代企业中的应用4.1 金融行业4.2 电子商务4.3 政府和公共服务4.4 医疗健康 结语 …