MyBatis级联探讨第一篇——概念和模型

谈谈MyBatis级联,文章尽量具体一些。
1、概述
关于MyBatis级联有三种:
association  :一对一的关联
collection   : 一对多的关联
discriminator :鉴别器,
它可以用于鉴别是使用那个resultMapper,或者你暂时不能理解,可以暂时放放。后面我们会再谈。
MyBatis当前只能支持查询的级联,而不能支持新增,删除和更新这些操作。所以这里只会讨论查询的级联。当然首先谈论的是三种级联的用法,然后就是N+1问题,延迟加载的问题。
2、模型  
在讨论这些关联之前,我们需要先定义一个模型。


我们先简单描述一下模型:
我们有一个员工表,一个员工卡表,显然员工和员工卡是一对一的关系。
而员工有男性员工和女性员工所以他们的健康表是不一样的。
一个员工可以参与多个工程,所以有张员工工程表把,员工和工程关联起来了,员工和工程是一对多的关系。
下面笔者给出建表语句:


CREATE TABLE t_employee 
(
   id                 INT(20)              NOT NULL AUTO_INCREMENT,
   emp_name           VARCHAR(60)          NOT NULL,
  sex                TINYINT(2)              NOT NULL,
   PRIMARY KEY (id)
);

CREATE TABLE t_employee_card 
(
   id                 INT(20)              NOT NULL AUTO_INCREMENT,
   emp_id             INT(20)              NOT NULL,
   card_no            VARCHAR(20)          NOT NULL,
   PRIMARY KEY (id)
);

CREATE TABLE t_employee_project 
(
   id                 INT(20)              NOT NULL AUTO_INCREMENT,
   emp_id             INT(20),
   proj_id             INT(20),
   PRIMARY KEY (id)
);

CREATE TABLE t_healthy_male 
(
   id                 INT(20)              NOT NULL AUTO_INCREMENT,
   emp_id             INT(20),
   prostate           VARCHAR(60),
   PRIMARY KEY (id)
);

CREATE TABLE t_healthy_remale 
(
   id                 INTEGER              NOT NULL AUTO_INCREMENT,
   emp_id             INT(20),
   uterus             VARCHAR(60),
   PRIMARY KEY (id)
);

CREATE TABLE t_project 
(
   id                 INT(20)              NOT NULL AUTO_INCREMENT,
   proj_name          VARCHAR(60),
   PRIMARY KEY (id)
);

3、定义POJO

让我们先定义POJO,如下:

public class Employee implements Serializable {


    private static final long serialVersionUID = 5892612078624312365L;


    private Integer id;
    private String empName;
    private Integer sex;
    //员工卡,一对一关联
    private EmployeeCard employeeCard;
    //项目,一对多关联
    private List<Project> projectList;
    //########setter and getter############
}


public class MaleEmployee extends Employee implements Serializable {




    private static final long serialVersionUID = 6597520360502006892L;




    private List<String> prostateList;


    //########setter and getter############
}


public class FemaleEmployee extends Employee implements Serializable {


    private static final long serialVersionUID = -7565810909492919315L7;

    private List<String> uterusList;

    //########setter and getter############
}


public class EmployeeCard implements Serializable {

    private static final long serialVersionUID = -7593225091990005041L;

    private Integer empId;
    private Integer cardNo;

    //########setter and getter############
}


public class Project implements Serializable {


    private static final long serialVersionUID = -7834541701010022018L;


    private Integer id;
    private String projName;
    //########setter and getter############
}

除了Employee外,都是很简单的POJO。Employee其实也不复杂,Employee是基类,然后通过其派生一个男性一个女性的类,也不难理解。


下面,我们将就这三种级联展开更深一层的讨论。


 




Mybatis级联查询可以通过在Mapper文件中使用嵌套查询的方式实现,具体步骤如下: 1. 在POJO类中定义关联属性(如一对多、多对多等),并提供对应的setter和getter方法。 2. 在Mapper文件中定义对应的嵌套查询语句,使用关联属性的getter方法来获取关联对象的数据,例如: ```xml <select id="findUserById" parameterType="int" resultMap="userResultMap"> select * from user where id = #{id} </select> <resultMap id="userResultMap" type="User"> <id column="id" property="id"/> <result column="username" property="username"/> <result column="age" property="age"/> <association property="department" javaType="Department"> <id column="dept_id" property="id"/> <result column="dept_name" property="name"/> </association> </resultMap> <select id="findUserWithDeptById" parameterType="int" resultMap="userResultMap"> select u.*, d.dept_name from user u left join department d on u.dept_id = d.id where u.id = #{id} </select> ``` 在上述代码中,`findUserById`只查询User表中的数据,而`findUserWithDeptById`则查询User表和Department表中的数据,并将Department作为User对象的关联属性返回。 3. 在业务层中调用Mapper接口的方法,即可进行级联查询,例如: ```java User user = userMapper.findUserWithDeptById(1); System.out.println(user.getDepartment().getName()); ``` 在上述代码中,`userMapper.findUserWithDeptById`方法会返回一个User对象,其中的Department属性已经被赋值为关联的Department对象,通过getDepartment()方法即可获取Department对象的数据。 需要注意的是,在进行级联查询时,需要定义好关联属性的类型和对应的嵌套查询语句,否则会导致查询失败或数据不完整。同时,级联查询也会增加数据库的查询开销,应该根据实际情况进行使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值