MyBatis中多對多關系的映射和查詢


先說一下需求:

  在頁面上顯示數據庫中的所有圖書,顯示圖書的同時,顯示出該圖書所屬的類別(這里一本書可能同時屬於多個類別)

 

測試環境:MySQL、MyEclipse

創建表:

  筆者這里使用 中間表 連接 圖書表 和 圖書類別表,圖書表中 沒有使用外鍵關聯 圖書類別表

  而是在中間表中引用了 圖書主鍵 和 類別主鍵 

  通過中間表來 表示 圖書 和 圖書類別 的關系

 

建立圖書表(圖書編號,圖書名字)

create table book
(
    bid int primary key auto_increment,
    bname varchar(20)                                     
);

建立類別表(類別編號,類別名字)

create table category
(
    cid int primary key auto_increment,
    cname varchar(20)
);

建立中間表(圖書編號,類別編號)

復制代碼
create table middle
(
    m_bid int,
    m_cid int,
    constraint fk_bid foreign key(m_bid) references book(bid),
    constraint fk_cid foreign key(m_cid) references category(cid)
);
復制代碼

插入測試數據

復制代碼
insert into category values (default,'java');
insert into category values (default,'c++');
insert into category values (default,'mysql');

insert into book values (default,'SQL技術');
insert into book values (default,'SSM+MySQL詳解');
insert into book values (default,'C++和java對比');

insert into middle values (1,3);
insert into middle values (2,1);
insert into middle values (2,3);
insert into middle values (3,2);
insert into middle values (3,1);
復制代碼

插入的數據中,第一本書 有一個類別,第二本書和第三本書都有兩個類別

到現在為止,數據庫的事情就完事了。下面,通過MyBatis-Generator生成實體類、DAO接口、XML映射文件  不會點擊這里

為了方便省事,筆者這里通過Java項目演示,將自動生成的文件 放入新建的Java項目中,導入相關的Jar包,項目結構 如下圖

現在我們打開生成的 圖書實體類 看一下

復制代碼
public class Book {
private Integer bid; private String bname; public Integer getBid() { return bid; } public void setBid(Integer bid) { this.bid = bid; } public String getBname() { return bname; } public void setBname(String bname) { this.bname = bname == null ? null : bname.trim(); } }
復制代碼

只有圖書編號、圖書名字 這兩個屬性,而我們的需求是 得到圖書的同時,得到該圖書所屬的 所有類別, 所以 我們可以考慮 給圖書實體類 添加 一個 圖書類別的集合

修改后的圖書實體類 如下

復制代碼
public class Book {
    private Integer bid;
    private String bname;
    private List<Category> categories;

    public Integer getBid() {
        return bid;
    }
    public void setBid(Integer bid) {
        this.bid = bid;
    }
    public String getBname() {
        return bname;
    }
    public void setBname(String bname) {
        this.bname = bname == null ? null : bname.trim();
    }
    public List<Category> getCategories() {
        return categories;
    }
    public void setCategories(List<Category> categories) {
        this.categories = categories;
    }
}
復制代碼

下面 我們開始寫SQL語句,使用連接查詢 查出所有的圖書和圖書類別

復制代碼
select 
    * 
from 
    book b
  inner join  
      middle m
  on
      b.bid=m.m_bid
  inner join
      category c
  on
      m.m_cid=c.cid
復制代碼

執行結果如下 完美的顯示了所有圖書 和 該圖書的類別

下面 我們就在XML映射文件中動手腳,使得 這些數據 能按我們所期望的 自動填充到 圖書實體類中

這里為突出重點 所以將圖書的映射文件和DAO接口 清空,清空后 如下

復制代碼
<?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.mybatis.dao.BookMapper" >
    <select id="queryAll">
        
    </select>
</mapper>
復制代碼
public interface BookMapper {
    
   List<Book> queryAll();
   
}

 

清空后 我們開始編寫,編寫結果如下

復制代碼
<?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.mybatis.dao.BookMapper" >
    
    <resultMap type="com.mybatis.entity.Book" id="bookMap">
        <id property="bid" column="bid" />
        <result property="bname" column="bname" />
        
        <collection  property="categories"  ofType="com.mybatis.entity.Category">
            <id property="cid" column="cid" />
            <result property="cname" column="cname" />    
        </collection>
    </resultMap>

    <select id="queryAll" resultMap="bookMap">
        select 
            * 
        from 
            book b
        inner join  
            middle m
        on
            b.bid=m.m_bid
        inner join
            category c
        on
            m.m_cid=c.cid
    </select>
</mapper>
復制代碼

最后我們 編寫main方法測試

復制代碼
public class MyMain {
    
    public static void main(String[] args) throws IOException {
        
         String resource = "mybatis-config.xml";
         Reader reader = Resources.getResourceAsReader(resource);
         SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(reader);
         SqlSession session = factory.openSession();
         
         BookMapper bookMapper = session.getMapper(BookMapper.class);
         for (Book book : bookMapper.queryAll()) {
            System.out.print("["+book.getBname()+"]");
            for(Category category :book.getCategories()){
                System.out.print(category.getCname()+"\t");
            }
            System.out.println("\n");
        }
    }
}
復制代碼

測試結果如下圖

成功輸出了 所有圖書 和 對應的圖書類別

完整項目下載:點擊下載 (為方便,筆者將SSM整合的jar包引入到項目中了,大家可以刪去無用的jar包)

筆者能力有限,有哪些可以改進或者不對的地方歡迎提出!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM