真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

MyBatis的嵌套查詢解析

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

成都創(chuàng)新互聯(lián)作為成都網(wǎng)站建設(shè)公司,專注重慶網(wǎng)站建設(shè)公司、網(wǎng)站設(shè)計,有關(guān)成都定制網(wǎng)頁設(shè)計方案、改版、費用等問題,行業(yè)涉及成都紗窗等多個領(lǐng)域,已為上千家企業(yè)服務(wù),得到了客戶的尊重與認(rèn)可。

以最簡單的用戶表訂單表這個最簡單的一對多做示例:

對應(yīng)的JavaBean:

User:

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

 }

User_orders:

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

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

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

嵌套語句查詢

mybatis提供了一種機制,叫做嵌套語句查詢,可以大大簡化上述的操作,加入配置及代碼如下:


 
 
  
  
 
 




測試(可以成功查詢到所有信息):

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

嵌套語句查詢的原理

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

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

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

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

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

嵌套查詢的多對一

上面的關(guān)聯(liá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:


 
 
  
    
   
  


 

  

測試:

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

嵌套查詢的N+1問題

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

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

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

嵌套結(jié)果查詢

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

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



 
  
   
   
  

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

  

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

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

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

對于關(guān)聯(lián)的結(jié)果查詢,如果是多對一的關(guān)系,則通過形如 進行配置,Mybatis會通過column屬性對應(yīng)的user_id 值去從內(nèi)存中取數(shù)據(jù),并且封裝成User_order對象;

如果是一對多的關(guān)系,就如User和User_order之間的關(guān)系,通過形如 進行配置,MyBatis通過 id去內(nèi)存中取User_orders對象,封裝成List;

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

以上是通過查詢User表所有信息來演示了一對多和多對一的映射對象處理。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。


分享名稱:MyBatis的嵌套查詢解析
當(dāng)前URL:http://weahome.cn/article/jhddhs.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部