溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

MyBatis的嵌套查詢(xún)解析

發(fā)布時(shí)間:2020-10-10 16:58:25 來(lái)源:腳本之家 閱讀:119 作者:不能說(shuō)的秘密go 欄目:編程語(yǔ)言

Mybatis表現(xiàn)關(guān)聯(lián)關(guān)系比hibernate簡(jiǎn)單,沒(méi)有分那么細(xì)致one-to-many、many-to-one、one-to-one。而是只有兩種association(一)、collection(多),表現(xiàn)很簡(jiǎn)潔。下面通過(guò)一個(gè)實(shí)例,來(lái)展示一下Mybatis對(duì)于常見(jiàn)的一對(duì)多和多對(duì)一關(guān)系復(fù)雜映射是怎樣處理的。

以最簡(jiǎn)單的用戶(hù)表訂單表這個(gè)最簡(jiǎn)單的一對(duì)多做示例:

對(duì)應(yīng)的JavaBean:

User:

public class User {
  private int id;
  private String name;
  private Double age;
 private List<User_orders> orders;
 // get set 省

 }

User_orders:

public class User_orders {
 private int id;
 private String name;
 // get set 省
}

對(duì)應(yīng)的數(shù)據(jù)庫(kù):

mysql> desc user;
+-------+-------------+------+-----+---------+----------------+
| Field | Type    | Null | Key | Default | Extra     |
+-------+-------------+------+-----+---------+----------------+
| id  | int(11)   | NO  | PRI | NULL  | auto_increment |
| name | varchar(20) | NO  |   | NULL  |        |
| age  | double   | YES |   | NULL  |        |
+-------+-------------+------+-----+---------+----------------+
3 rows in set (0.00 sec)

mysql> desc user_orders;
+---------+-------------+------+-----+---------+----------------+
| Field  | Type    | Null | Key | Default | Extra     |
+---------+-------------+------+-----+---------+----------------+
| id   | int(11)   | NO  | PRI | NULL  | auto_increment |
| name  | varchar(20) | NO  |   | NULL  |        |
| user_id | int(5)   | YES | MUL | NULL  |        |
+---------+-------------+------+-----+---------+----------------+
3 rows in set (0.00 sec)

現(xiàn)在查詢(xún)一個(gè)user的id查詢(xún)出所有信息.如果不考慮關(guān)聯(lián)查詢(xún),我們會(huì)先根據(jù)user的id在user表中查詢(xún)出name,age然后設(shè)置給User類(lèi)的時(shí)候,再根據(jù)該user的id在user_orders表中查詢(xún)出所有訂單并設(shè)置給User類(lèi)。這樣的話(huà),在底層最起碼調(diào)用兩次查詢(xún)語(yǔ)句,得到需要的信息,然后再組裝User對(duì)象。

嵌套語(yǔ)句查詢(xún)

mybatis提供了一種機(jī)制,叫做嵌套語(yǔ)句查詢(xún),可以大大簡(jiǎn)化上述的操作,加入配置及代碼如下:

<resultMap type="domain.User" id="user">
 <id column="id" property="id"/>
 <result column="age" property="age"/>
 <collection column="id" property="orders" ofType="domain.User_orders"
  select="selectOrderByUser"> 
  <id column="id" property="id"/>
 <result column="name" property="name"/>
 </collection>
</resultMap>
<select id="selectOrderByUser" parameterType="integer" resultType="domain.User_orders">
  select id,name from user_orders where user_id = #{id}
</select>

<select id="findById" resultMap="user" parameterType="integer">
     select * from user where id = #{id}
  </select>

測(cè)試(可以成功查詢(xún)到所有信息):

String config = "sqlMapConfig.xml";
InputStream inputStream = Resources.getResourceAsStream(config);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession session = sqlSessionFactory.openSession();
// 執(zhí)行在bean配置文件中定義的sql語(yǔ)句
User user = session.selectOne("UserMapper.findById", 1);
//一句即可獲取到復(fù)雜的User對(duì)象。
System.out.println(user);
session.commit();
session.close();

嵌套語(yǔ)句查詢(xún)的原理

在上面的代碼中,Mybatis會(huì)執(zhí)行以下流程:

1.先執(zhí)行 findById 對(duì)應(yīng)的語(yǔ)句從User表里獲取到ResultSet結(jié)果集;

2.取出ResultSet下一條有效記錄,然后根據(jù)resultMap定義的映射規(guī)格,通過(guò)這條記錄的數(shù)據(jù)來(lái)構(gòu)建對(duì)應(yīng)的一個(gè)User 對(duì)象。

當(dāng)要對(duì)User中的orders屬性進(jìn)行賦值的時(shí)候,發(fā)現(xiàn)有一個(gè)關(guān)聯(lián)的查詢(xún),此時(shí)Mybatis會(huì)先執(zhí)行這個(gè)select查詢(xún)語(yǔ)句,得到返回的結(jié)果,將結(jié)果設(shè)置到user的orders屬性上

這種關(guān)聯(lián)的嵌套查詢(xún),有一個(gè)非常好的作用就是:可以重用select語(yǔ)句,通過(guò)簡(jiǎn)單的select語(yǔ)句之間的組合來(lái)構(gòu)造復(fù)雜的對(duì)象。想如上的兩個(gè)select完全可以獨(dú)立使用。

嵌套查詢(xún)的多對(duì)一

上面的關(guān)聯(lián)查詢(xún)查詢(xún)其實(shí)是對(duì)于一對(duì)多的查詢(xún),即從user中查出user_order的信息。

現(xiàn)在從user_order中查user的信息.

在User_order表中增加字段user:

public class User_orders {
 private int id;
 private String name;
 private User user;
 //xxx
}

配置select:

<resultMap type="domain.User_orders" id="user_order">
 <id column="id" property="id"/>
 <result column="name" property="name"/>
  <association property="user" column="user_id" javaType="domain.User" select="selectUserByOrderId">
    <id column="id" property="id"/>
   <result column="age" property="age"/>
  </association>
</resultMap>

 <select id="selectUserByOrderId" parameterType="INTEGER" resultType="domain.User">
   select id,age from user where id = #{id}
 </select>

  <select id="findOne" resultMap="user_order" parameterType="integer">
    select * from user_orders where id=#{id}
  </select>

測(cè)試:

SqlSession session = sqlSessionFactory.openSession();
    // 執(zhí)行在bean配置文件中定義的sql語(yǔ)句
    User_orders user_orders= session.selectOne("User_ordersMapper.findOne", 1);
    System.out.println(user_orders);
    //查詢(xún)到了user_order對(duì)應(yīng)的user的信息
    session.commit();
    session.close();

嵌套查詢(xún)的N+1問(wèn)題

盡管嵌套查詢(xún)大量的簡(jiǎn)化了存在關(guān)聯(lián)關(guān)系的查詢(xún),但它的弊端也比較明顯:即所謂的N+1問(wèn)題。關(guān)聯(lián)的嵌套查詢(xún)顯示得到一個(gè)結(jié)果集,然后根據(jù)這個(gè)結(jié)果集的每一條記錄進(jìn)行關(guān)聯(lián)查詢(xún)。

現(xiàn)在假設(shè)嵌套查詢(xún)就一個(gè)(即resultMap 內(nèi)部就一個(gè)association標(biāo)簽),現(xiàn)查詢(xún)的結(jié)果集返回條數(shù)為N,那么關(guān)聯(lián)查詢(xún)語(yǔ)句將會(huì)被執(zhí)行N次,加上自身返回結(jié)果集查詢(xún)1次,共需要訪(fǎng)問(wèn)數(shù)據(jù)庫(kù)N+1次。如果N比較大的話(huà),這樣的數(shù)據(jù)庫(kù)訪(fǎng)問(wèn)消耗是非常大的!所以使用這種嵌套語(yǔ)句查詢(xún)的使用者一定要考慮慎重考慮,確保N值不會(huì)很大。

以上面一對(duì)多(根據(jù)user的id查詢(xún)order)的例子為例,select 語(yǔ)句本身會(huì)返回user條數(shù)為1 的結(jié)果集,由于它存在有1條關(guān)聯(lián)的語(yǔ)句查詢(xún),它需要共訪(fǎng)問(wèn)數(shù)據(jù)庫(kù) 1*(1+1)=2次數(shù)據(jù)庫(kù)。

嵌套結(jié)果查詢(xún)

嵌套語(yǔ)句的查詢(xún)會(huì)導(dǎo)致數(shù)據(jù)庫(kù)訪(fǎng)問(wèn)次數(shù)不定,進(jìn)而有可能影響到性能。Mybatis還支持一種嵌套結(jié)果的查詢(xún):即對(duì)于一對(duì)多,多對(duì)多,多對(duì)一的情況的查詢(xún),Mybatis通過(guò)聯(lián)合查詢(xún),將結(jié)果從數(shù)據(jù)庫(kù)內(nèi)一次性查出來(lái),然后根據(jù)其一對(duì)多,多對(duì)一,多對(duì)多的關(guān)系和ResultMap中的配置,進(jìn)行結(jié)果的轉(zhuǎn)換,構(gòu)建需要的對(duì)象。

重新定義User的結(jié)果映射 resultMap

<resultMap type="domain.User" id="user_auto">
<id column="id" property="id"/>
 <result column="age" property="age"/>
 <collection column="id" property="orders" ofType="domain.User_orders"> 
   <id column="order_id" property="id"/>
   <result column="name" property="name"/>
  </collection>
</resultMap>

對(duì)應(yīng)的sql語(yǔ)句如下:

  <select id="findAuth" resultMap="user_auto">
  select u.id,u.age,o.id as order_id ,o.name,o.user_id as user_id from user u left outer join user_orders o
  on o.user_id = u.id
  </select>

嵌套結(jié)果查詢(xún)的執(zhí)行步驟:

1.根據(jù)表的對(duì)應(yīng)關(guān)系,進(jìn)行join操作,獲取到結(jié)果集;

  1. 根據(jù)結(jié)果集的信息和user 的resultMap定義信息,對(duì)返回的結(jié)果集在內(nèi)存中進(jìn)行組裝、賦值,構(gòu)造User;
  2. 返回構(gòu)造出來(lái)的結(jié)果List 結(jié)果。

對(duì)于關(guān)聯(lián)的結(jié)果查詢(xún),如果是多對(duì)一的關(guān)系,則通過(guò)形如 <association property="user" column="user_id" javaType="domain.User" > 進(jìn)行配置,Mybatis會(huì)通過(guò)column屬性對(duì)應(yīng)的user_id 值去從內(nèi)存中取數(shù)據(jù),并且封裝成User_order對(duì)象;

如果是一對(duì)多的關(guān)系,就如User和User_order之間的關(guān)系,通過(guò)形如 <collection column="id" property="orders" ofType="domain.User_orders">進(jìn)行配置,MyBatis通過(guò) id去內(nèi)存中取User_orders對(duì)象,封裝成List;

對(duì)于關(guān)聯(lián)結(jié)果的查詢(xún),只需要查詢(xún)數(shù)據(jù)庫(kù)一次,然后對(duì)結(jié)果的整合和組裝全部放在了內(nèi)存中。

以上是通過(guò)查詢(xún)User表所有信息來(lái)演示了一對(duì)多和多對(duì)一的映射對(duì)象處理。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI