Mybatis學(xué)習(xí)筆記(四):映射器介紹(下)

本文檔講述MyBatis映射器較為高級(jí)的部分撑柔。

級(jí)聯(lián)

級(jí)聯(lián)是resultMap中的配置获询,用來(lái)實(shí)現(xiàn)一對(duì)一或一對(duì)多的連表查詢掖棉。

級(jí)聯(lián)不是必須的聂抢,使用級(jí)聯(lián)的好處是獲取關(guān)聯(lián)數(shù)據(jù)十分便捷拯钻,但是級(jí)聯(lián)過(guò)多會(huì)增加系統(tǒng)的復(fù)雜度帖努,降低系統(tǒng)的性能。因此當(dāng)層級(jí)超過(guò)3層是粪般,就不再考慮使用級(jí)聯(lián)拼余。

MyBatis中的級(jí)聯(lián)分為3種:

  • 鑒別器(discriminator):它是一個(gè)根據(jù)某些條件決定采用具體實(shí)現(xiàn)類級(jí)聯(lián)的方案。
  • 一對(duì)一(association):一對(duì)一的級(jí)聯(lián)亩歹。
  • 一對(duì)多(collection): 一對(duì)多的級(jí)聯(lián)匙监。

先給出例子。創(chuàng)建3張表:教師表小作、班級(jí)表和學(xué)生表亭姥。假設(shè)一個(gè)班只有一個(gè)負(fù)責(zé)老師,但有多個(gè)學(xué)生顾稀。那么班級(jí)和老師就是一對(duì)一达罗,班級(jí)和學(xué)生就是一對(duì)多。

CREATE TABLE t_teachers(
 t_id INT PRIMARY KEY AUTO_INCREMENT, 
 t_name VARCHAR(20)
)ENGINE=INNODB DEFAULT CHARSET=utf8;
CREATE TABLE t_classes(
 c_id INT PRIMARY KEY AUTO_INCREMENT, 
 c_name VARCHAR(20), 
 teacher_id INT
)ENGINE=INNODB DEFAULT CHARSET=utf8;
ALTER TABLE t_class ADD CONSTRAINT fk_teacher_id FOREIGN KEY (teacher_id) REFERENCES t_teacher(t_id);    
CREATE TABLE t_students(
 s_id INT PRIMARY KEY AUTO_INCREMENT, 
 s_name VARCHAR(20), 
 class_id INT
)ENGINE=INNODB DEFAULT CHARSET=utf8;

在表里面插入幾條數(shù)據(jù)

INSERT INTO t_teacher(t_name) VALUES('張老師');
INSERT INTO t_teacher(t_name) VALUES('蔡老師');

INSERT INTO t_class(c_name, teacher_id) VALUES('318班', 1);
INSERT INTO t_class(c_name, teacher_id) VALUES('319班', 2);
INSERT INTO t_students(s_name, class_id) VALUES('張三', 1);
INSERT INTO t_students(s_name, class_id) VALUES('李四', 1);
INSERT INTO t_students(s_name, class_id) VALUES('小明', 1);
INSERT INTO t_students(s_name, class_id) VALUES('王五', 2);
INSERT INTO t_students(s_name, class_id) VALUES('小紅', 2);

創(chuàng)建實(shí)體類Teacher静秆,Student和ClassInfo粮揉。

package com.wyk.mybatisDemo.pojo;

public class Teacher {

    private int id;
    private String name;
    
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    
    @Override
    public String toString() {
        return "Teacher [id=" + id + ", name=" + name + "]";
    }
}
package com.wyk.mybatisDemo.pojo;

public class Student {

    private int id;
    private String name;
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    
    public String toString() {
        return "Student[id=" + id + ", name=" + name + "]";
    }
}
package com.wyk.mybatisDemo.pojo;

import java.util.List;

public class ClassInfo {

    private int id;
    private String name;
    private Teacher teacher;
    private List<Student> students;
    
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Teacher getTeacher() {
        return teacher;
    }
    public void setTeacher(Teacher teacher) {
        this.teacher = teacher;
    }
    

    public List<Student> getStudent() {
        return students;
    }
    public void setStudent(List<Student> students) {
        this.students = students;
    }
    public String toString() {
        return "ClassInfo [id=" + id + ", name=" + name + ", teacher=" + teacher + 
                ", students=" + students +"]";
    }
}

可以看到,ClassInfo類中包含一個(gè)Teacher對(duì)象和一個(gè)Student的List诡宗。定義查詢ClassInfo信息的方法滔蝉,有2種方式,先創(chuàng)建接口塔沃。

package com.wyk.mybatisDemo.mapper;

import com.wyk.mybatisDemo.pojo.ClassInfo;

public interface SchoolMapper {

    public ClassInfo getClassInfo(int id);
    
    public ClassInfo getClassInfo2(int id);
}

一種方法是通過(guò)聯(lián)表查詢獲取結(jié)果蝠引,另一種方法是多次查詢阳谍。具體方法請(qǐng)看映射器文件。

<?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.wyk.mybatisDemo.mapper.SchoolMapper">
    <resultMap type="com.wyk.mybatisDemo.pojo.ClassInfo" id="classMap">
        <id property="id" column="c_id" />
        <result property="name" column="c_name" />
        <association property="teacher" javaType="com.wyk.mybatisDemo.pojo.Teacher">
            <id property="id" column="t_id" />
            <result property="name" column="t_name" />
        </association>
        <collection property="students" ofType="com.wyk.mybatisDemo.pojo.Student">
            <id property="id" column="s_id" />
            <result property="name" column="s_name" />
        </collection>
    </resultMap>
    <!-- 方法一 聯(lián)表查詢 -->
    <select id="getClassInfo" parameterType="int" resultMap="classMap">
        select c.c_id, c.c_name, t.t_id, t.t_name , s.s_id, s.s_name
        from t_classes c, t_teachers t, t_students s
        where c.teacher_id = t.t_id and c.c_id = s.class_id and c.c_id=#{id} 
    </select>
    <resultMap type="com.wyk.mybatisDemo.pojo.ClassInfo" id="classMap2">
        <id property="id" column="c_id" />
        <result property="name" column="c_name" />
        <association property="teacher" column="teacher_id" select="getTeacher" />
        <collection property="students" ofType="com.wyk.mybatisDemo.pojo.Student"
            column="c_id" select="getStudent"></collection>
    </resultMap>
    <!-- 方法二 多次查詢 -->
    <select id="getClassInfo2" parameterType="int" resultMap="classMap2">
        select c_id, c_name, teacher_id from t_classes where c_id=#{id}
    </select>
    <select id="getTeacher" parameterType="int" resultType="com.wyk.mybatisDemo.pojo.Teacher">
        select t_id id, t_name name from t_teachers where t_id=#{id}
    </select>
    <select id="getStudent" parameterType="int" resultType="com.wyk.mybatisDemo.pojo.Student">
        select s_id id, s_name name from t_students where class_id=#{id}
    </select>
</mapper>

在單元測(cè)試中運(yùn)行一下螃概,查看效果矫夯。

@Test
public void testGetClassInfo() {
    SqlSession sqlSession = DataConnection.openSqlSession();
    SchoolMapper schoolMapper = sqlSession.getMapper(SchoolMapper.class);
    ClassInfo classInfo = schoolMapper.getClassInfo(1);
    System.out.println(classInfo);
    ClassInfo classInfo2 = schoolMapper.getClassInfo2(1);
    System.out.println(classInfo2);
}
級(jí)聯(lián)結(jié)果.png

鑒別器的使用在實(shí)際中用的較少,就沒(méi)有寫(xiě)吊洼,主要是根據(jù)某些條件決定采用哪個(gè)類训貌。

延遲加載

關(guān)注上一小節(jié)的方式二,每當(dāng)查詢一個(gè)班級(jí)信息冒窍,都要查詢N個(gè)學(xué)生信息递沪,會(huì)造成很大的資源浪費(fèi),尤其是我們并不想知道學(xué)生信息的時(shí)候综液。

為了應(yīng)對(duì)上述的N+1問(wèn)題款慨,MyBatis提供了延遲加載功能。settings配置中有2個(gè)元素可以配置級(jí)聯(lián)谬莹。

  • lazyLoadingEnabled檩奠,延遲加載的全局開(kāi)關(guān),默認(rèn)為false附帽。
  • aggressiveLazyLoading, 層級(jí)延遲加載開(kāi)關(guān)埠戳,處于同一個(gè)層級(jí)的關(guān)聯(lián)表會(huì)同時(shí)延遲加載,或者同時(shí)被加載蕉扮。

使用延遲加載首先要引用對(duì)應(yīng)的jar包c(diǎn)glib整胃。

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>${mybatis.version}</version>
</dependency>
<settings>
    <setting name="lazyLoadingEnabled" value="true" />
    <setting name="aggressiveLazyLoading" value="true" />
</settings>

上面的配置屬于全局配置,會(huì)出現(xiàn)一個(gè)問(wèn)題慢显,同一個(gè)層級(jí)的級(jí)聯(lián)表也存在需求性的差異爪模,同一級(jí)的某個(gè)數(shù)據(jù)庫(kù)表的數(shù)據(jù)或許不是我們經(jīng)常使用的,那么上述的配置也會(huì)影響系統(tǒng)的性能荚藻。

fetchType屬性可以處理全局定義無(wú)法處理的問(wèn)題屋灌,進(jìn)行自定義,出現(xiàn)在級(jí)聯(lián)元素association和collection中应狱,有2個(gè)可選值:

  • eager共郭,獲取當(dāng)前實(shí)體后立即加載對(duì)應(yīng)的數(shù)據(jù)。
  • lazy疾呻,獲取當(dāng)前實(shí)體后延遲加載對(duì)應(yīng)的數(shù)據(jù)除嘹。
<collection property="students" ofType="com.wyk.mybatisDemo.pojo.Student" fetchType="eager">
        <id property="id" column="s_id" />
        <result property="name" column="s_name" />
    </collection>

fetchType屬性會(huì)忽略全局配置項(xiàng)lazyLoadingEnabled和aggressiveLazyLoading。

說(shuō)了這么多岸蜗,感覺(jué)最好還是使用上一小節(jié)的方法一連接查詢尉咕,而不是方法二嵌套查詢比較好。

緩存

MyBatis中擁有一級(jí)緩存和二級(jí)緩存璃岳,其中一級(jí)緩存是在SqlSession上的緩存年缎,二級(jí)緩存是在SqlSessionFactory上的緩存悔捶。

一級(jí)緩存

MyBatis默認(rèn)開(kāi)啟一級(jí)緩存,而無(wú)需進(jìn)行任何配置单芜。

還是上面的例子蜕该,編寫(xiě)一個(gè)查找的測(cè)試用例。

@Test
public void testFirstLevelCache() {
    SqlSession sqlSession = null;
    try {
        sqlSession = DataConnection.openSqlSession();
        SchoolMapper schoolMapper = sqlSession.getMapper(SchoolMapper.class);
        ClassInfo classInfo1= schoolMapper.getClassInfo2(1);
        logger.info("再獲取一次實(shí)體...");
        ClassInfo classInfo2 = schoolMapper.getClassInfo2(1);
    } catch (Exception e) {
        logger.info(e.getMessage(), e);
    } finally {
        if(sqlSession != null) {
            sqlSession.close();
        }
    }   
}

例子里進(jìn)行了兩次查詢洲鸠,查看控制臺(tái)堂淡,發(fā)現(xiàn)只執(zhí)行了一次SQL語(yǔ)句。

一級(jí)緩存結(jié)果.png

每次查詢結(jié)果MyBatis都會(huì)進(jìn)行緩存扒腕,因此第二次查詢的時(shí)候可以直接獲取結(jié)果绢淀。但是如果執(zhí)行了增、刪袜匿、改等操作更啄,則緩存會(huì)刷新稚疹。

二級(jí)緩存

將測(cè)試用例改為創(chuàng)建2個(gè)sqlSessoin,如下所示居灯。

@Test
public void testSecondLevelCache() {
    SqlSession sqlSession1 = null;
    SqlSession sqlSession2 = null;
    try {
        sqlSession1 = DataConnection.openSqlSession();
        SchoolMapper schoolMapper1 = sqlSession1.getMapper(SchoolMapper.class);
        ClassInfo classInfo1= schoolMapper1.getClassInfo2(1);
        //需要提交,MyBatis才會(huì)緩存對(duì)象到SqlSessionFactory
        sqlSession1.commit();
        logger.info("再獲取一次實(shí)體...");
        sqlSession2 = DataConnection.openSqlSession();
        SchoolMapper schoolMapper2 = sqlSession2.getMapper(SchoolMapper.class);
        ClassInfo classInfo2= schoolMapper2.getClassInfo2(1);
        sqlSession2.commit();
    } catch (Exception e) {
        logger.info(e.getMessage(), e);
    } finally {
        if(sqlSession1 != null) {
            sqlSession1.close();
        }
        if(sqlSession2 != null) {
            sqlSession2.close();
        }
    }   
}

commit方法是為了將方法提交到SqlSessionFactory内狗。查看結(jié)果怪嫌,發(fā)現(xiàn)進(jìn)行了兩次查詢。

二級(jí)緩存測(cè)試結(jié)果1.png

在映射器文件中添加二級(jí)緩存配置cache柳沙。

<cache />

再次運(yùn)行用例岩灭,查看結(jié)果,發(fā)現(xiàn)命中緩存,只查詢了一次赂鲤。

二級(jí)緩存測(cè)試結(jié)果2.png

緩存配置項(xiàng)

前面是通用配置噪径,還可以為每個(gè)語(yǔ)句單獨(dú)配置。下面是各個(gè)語(yǔ)句的默認(rèn)配置数初,可以根據(jù)實(shí)際需求修改找爱。

<select ... flushCache="false" useCache="true" />
<insert ... flushCache="true" />
<update ... flushCache="true" />
<delete ... flushCache="true" />

另外,MyBatis還可以使用Redis等外部緩存泡孩。

存儲(chǔ)過(guò)程

待補(bǔ)充车摄。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市仑鸥,隨后出現(xiàn)的幾起案子吮播,更是在濱河造成了極大的恐慌,老刑警劉巖眼俊,帶你破解...
    沈念sama閱讀 219,039評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件意狠,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡疮胖,警方通過(guò)查閱死者的電腦和手機(jī)环戈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)誊役,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人谷市,你說(shuō)我怎么就攤上這事蛔垢。” “怎么了迫悠?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,417評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵鹏漆,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我创泄,道長(zhǎng)艺玲,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,868評(píng)論 1 295
  • 正文 為了忘掉前任鞠抑,我火速辦了婚禮饭聚,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘搁拙。我一直安慰自己秒梳,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,892評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布箕速。 她就那樣靜靜地躺著酪碘,像睡著了一般。 火紅的嫁衣襯著肌膚如雪盐茎。 梳的紋絲不亂的頭發(fā)上兴垦,一...
    開(kāi)封第一講書(shū)人閱讀 51,692評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音字柠,去河邊找鬼探越。 笑死,一個(gè)胖子當(dāng)著我的面吹牛窑业,可吹牛的內(nèi)容都是我干的钦幔。 我是一名探鬼主播,決...
    沈念sama閱讀 40,416評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼数冬,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼节槐!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起拐纱,我...
    開(kāi)封第一講書(shū)人閱讀 39,326評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤铜异,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后秸架,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體揍庄,經(jīng)...
    沈念sama閱讀 45,782評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,957評(píng)論 3 337
  • 正文 我和宋清朗相戀三年东抹,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蚂子。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片沃测。...
    茶點(diǎn)故事閱讀 40,102評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖食茎,靈堂內(nèi)的尸體忽然破棺而出蒂破,到底是詐尸還是另有隱情,我是刑警寧澤别渔,帶...
    沈念sama閱讀 35,790評(píng)論 5 346
  • 正文 年R本政府宣布附迷,位于F島的核電站,受9級(jí)特大地震影響哎媚,放射性物質(zhì)發(fā)生泄漏喇伯。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,442評(píng)論 3 331
  • 文/蒙蒙 一拨与、第九天 我趴在偏房一處隱蔽的房頂上張望稻据。 院中可真熱鬧,春花似錦买喧、人聲如沸捻悯。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,996評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)秋度。三九已至,卻和暖如春钱床,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背埠居。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,113評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工查牌, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人滥壕。 一個(gè)月前我還...
    沈念sama閱讀 48,332評(píng)論 3 373
  • 正文 我出身青樓纸颜,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親绎橘。 傳聞我的和親對(duì)象是個(gè)殘疾皇子胁孙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,044評(píng)論 2 355