mybatis一对多关联查询

一对多关联查询指的是在查询一方对象的时候同时把跟他所关联的多方对象也查询出来,这里以篮球队和篮球运动员为例,一个篮球队关联着多个篮球队员。

创建数据库表

创建球队的表,里面有两个字段:
id:主键
name:球队名称

CREATE TABLE `t_team` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `name` VARCHAR(45) NULL,
  PRIMARY KEY (`id`));

创建球员的表,有三个字段:
id:主键
name:球员姓名
tid:所在球队的主键id

CREATE TABLE `t_player` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `name` VARCHAR(45) NULL,
  `tid` INT NULL,
  PRIMARY KEY (`id`));

创建javabean

创建篮球运动员的类Player:

package com.monkey1024.bean;

/**
 * 篮球队员
 */
public class Player {

    private int id;

    private String name;

    public Player(String name) {
        this.name = name;
    }

    public Player() {
    }

     @Override
    public String toString() {
        return "Player{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }

    //省略setter和getter
}

创建球队的类Team,这里将球员数据全部放到一个List集合中:

package com.monkey1024.bean;

import java.util.List;

/**
 * 篮球队
 */
public class Team {

    private int id;

    private String name;

    private List<Player> playerList;


    @Override
    public String toString() {
        return "Team{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", playerList=" + playerList +
                '}';
    }

    //省略setter和getter
}

需要注意的是若定义的类是双向关联,即双方的属性中均有对方对象作为域属性出现,那么它们在定义各自的 toString()方法时需要注意,只让某一方可以输出另一方即可,不要让双方的 toString()方法均可输出对方。这样会造成栈内存溢出的错误。

创建TeamMapper.xml映射文件:

此时涉及到两张表的操作,即使你的字段名和bean属性名一致也要编写resultMap来进行关联。

<?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">
<mapper namespace="com.monkey1024.dao.TeamDao">

    <resultMap id="teamMap" type="Team">
        <!--team中的基本属性-->
        <id column="teamid" property="id"/>
        <result column="tname" property="name"/>

        <!--关联属性的映射关系-->
        <collection property="playerList" ofType="Player">
            <id column="pid" property="id"/>
            <result column="pname" property="name"/>
        </collection>
    </resultMap>


    <select id="selectTeamById" resultMap="teamMap">
        SELECT t.id teamid,t.name tname,p.id pid,p.name pname
        FROM t_team t , t_player p
        WHERE t.id = p.tid and t.id=#{id}
    </select>
</mapper>

上面sql语句中的collection标签可以表示集合类型,其中:

  • property中填写集合在Team这个bean中的属性名称,此时是playerList
  • ofType中填写集合的泛型

在collection标签中填写的内容关联的是Player中的属性。

创建dao

这里创建一个TeamDao的接口,里面添加一个方法:

package com.monkey1024.dao;

import com.monkey1024.bean.Team;

public interface TeamDao {

    Team selectTeamById(int id);
}

创建测试类

创建一个测试类用来执行上面的sql语句,注意不要忘了SqlSession和TeamDao对象的获取:

package com.monkey1024.test;

import com.monkey1024.bean.Team;
import com.monkey1024.dao.StudentDao;
import com.monkey1024.dao.TeamDao;
import com.monkey1024.util.MyBatisUtil;
import org.apache.ibatis.session.SqlSession;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

public class TeamTest01 {

    private SqlSession sqlSession;

    private TeamDao teamDao;

    /**
     * 测试时先执行该方法创建StudentDao对象
     */
    @Before
    public void initStudentDao(){
        sqlSession = MyBatisUtil.getSqlSession();
        //通过该方法可以获取TeamDao的对象
        teamDao = sqlSession.getMapper(TeamDao.class);
    }

    /**
     * 执行完成后需要关闭sqlSession
     */
    @After
    public void closeSession() {
        if (sqlSession != null) {
            sqlSession.close();
        }
    }

    @Test
    public void selectTeamById() {
        Team team = teamDao.selectTeamById(1);

        System.out.println(team);
    }

}

在执行成功后,可以看到查询出的数据只有一个Team对象,在该对象中的list属性里面存放了多个Player对象数据。