欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

mybatis 集合嵌套查詢和集合嵌套結(jié)果的區(qū)別說(shuō)明

 更新時(shí)間:2021年09月23日 16:23:05   作者:-夢(mèng)與時(shí)光遇-  
這篇文章主要介紹了mybatis 集合嵌套查詢和集合嵌套結(jié)果的區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

集合嵌套查詢和集合嵌套結(jié)果的區(qū)別

嵌套查詢是多條sql語(yǔ)句分開寫并配置,嵌套結(jié)果是一條sql語(yǔ)句關(guān)聯(lián)查詢并配置,實(shí)質(zhì)效果是一樣的。嵌套語(yǔ)句的查詢會(huì)導(dǎo)致數(shù)據(jù)庫(kù)訪問次數(shù)不定,進(jìn)而有可能影響到性能。

1.創(chuàng)建2張表,建立主外鍵關(guān)系

在這里插入圖片描述

2.建立實(shí)體類

package com.yw.test06;  
public class Class  
{  
    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 "Class [id=" + id + ", name=" + name + "]";  
    }  
}  


package com.yw.test06;    
public class Student  
{  
    private int id;  
    private String name;  
    private int age;  
    private Class c;  
      
    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 int getAge()  
    {  
        return age;  
    }  
    public void setAge(int age)  
    {  
        this.age = age;  
    }  
    public Class getC()  
    {  
        return c;  
    }  
    public void setC(Class c)  
    {  
        this.c = c;  
    }  
    @Override  
    public String toString()  
    {  
        return "Student [id=" + id + ", name=" + name + ", age=" + age + ", c=" + c + "]";  
    } 
}  

3.修改配置文件

<?xml version="1.0" encoding="UTF-8" ?>  
<!DOCTYPE configuration  
  PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  
  "http://mybatis.org/dtd/mybatis-3-config.dtd">  
<configuration>  
    <properties resource="config.properties">  
    </properties>  
    <typeAliases>  
        <!-- <typeAlias type="com.yw.test06.StudentMapper" alias="Student" />  
        <typeAlias type="com.yw.test06.ClassMapper" alias="Class" /> -->  
        <package name="com.yw.test06"/>  
    </typeAliases>  
    <environments default="development">  
        <environment id="development">  
            <transactionManager type="JDBC" />  
            <dataSource type="POOLED">  
                <property name="driver" value="${driver}" />  
                <property name="url" value="${url}" />  
                <property name="username" value="${username}" />  
                <property name="password" value="${password}" />  
            </dataSource>  
        </environment>  
    </environments>  
    <mappers>  
        <!-- <mapper resource="org/mybatis/example/BlogMapper.xml"/> -->  
        <mapper resource="com/yw/test06/StudentMapper.xml" />  
    </mappers>  
  
</configuration>  

4.建立映射文件

1)嵌套查詢

<?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.yw.test07.StudentMapper">  
  
    <select id="selectStudent" resultMap="studentResult">  
        SELECT * FROM student s  
        WHERE s.ID = #{id}  
    </select>  
    <resultMap type="Student" id="studentResult">  
        <association property="c" column="c_id" javaType="Class"  
            select="selectClass" />  
    </resultMap>  
  
    <select id="selectClass" resultType="Class">  
        SELECT * FROM class WHERE ID= #{id}  
    </select>  
  
</mapper>  

2)嵌套結(jié)果

<?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.yw.test06.StudentMapper">  
    <resultMap id="studentResult" type="com.yw.test06.Student">  
        <id property="id" column="id"/>  
        <result property="name"  column="name"/>  
        <result property="age"  column="age"/> 
        <association property="c" resultMap="classResult"  
            javaType="Class"></association>  
  
    </resultMap>  
    <resultMap type="com.yw.test06.Class" id="classResult">  
        <id property="id" column="id"/>  
        <result property="name" column="name"/>  
    </resultMap>  
   
    <select id="selectStudent" resultMap="studentResult">   
        SELECT s.id,c.id,s.name,s.age,c.name from student s  left join class c on c.id=s.c_id where s.id=#{id}  
    </select>  
</mapper>  

5.創(chuàng)建測(cè)試類

package com.yw.test06;    
import java.io.IOException;  
import java.io.InputStream; 
import org.apache.ibatis.io.Resources;  
import org.apache.ibatis.session.SqlSession;  
import org.apache.ibatis.session.SqlSessionFactory;  
import org.apache.ibatis.session.SqlSessionFactoryBuilder;  
public class Test01  
{  
    public static void main(String[] args) throws IOException  
    {  
          
        String resource = "com/yw/test06/mybatis-config.xml";  
        InputStream inputStream = Resources.getResourceAsStream(resource);  
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);  
          
        SqlSession session = sqlSessionFactory.openSession(false);  
        try {  
          Student user = (Student) session.selectOne("com.yw.test06.StudentMapper.selectStudent", 1);  
          System.out.println(user);           
              
        } finally {  
          session.close();  
        }  
    }  
}  

MyBatis 嵌套查詢解析

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

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

嵌套語(yǔ)句查詢

mybatis提供了一種機(jī)制,叫做嵌套語(yǔ)句查詢,可以大大簡(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è)試(可以成功查詢到所有信息):

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ǔ)句查詢的原理

在上面的代碼中,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)的查詢,此時(shí)Mybatis會(huì)先執(zhí)行這個(gè)select查詢語(yǔ)句,得到返回的結(jié)果,將結(jié)果設(shè)置到user的orders屬性上這種關(guān)聯(lián)的嵌套查詢,有一個(gè)非常好的作用就是:可以重用select語(yǔ)句,通過(guò)簡(jiǎn)單的select語(yǔ)句之間的組合來(lái)構(gòu)造復(fù)雜的對(duì)象。想如上的兩個(gè)select完全可以獨(dú)立使用。

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

上面的關(guān)聯(lián)查詢查詢其實(shí)是對(duì)于一對(duì)多的查詢,即從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);
        //查詢到了user_order對(duì)應(yīng)的user的信息
        session.commit();
        session.close();

嵌套查詢的N+1問題

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

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

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

嵌套結(jié)果查詢

嵌套語(yǔ)句的查詢會(huì)導(dǎo)致數(shù)據(jù)庫(kù)訪問次數(shù)不定,進(jìn)而有可能影響到性能。Mybatis還支持一種嵌套結(jié)果的查詢:即對(duì)于一對(duì)多,多對(duì)多,多對(duì)一的情況的查詢,Mybatis通過(guò)聯(liá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é)果查詢的執(zhí)行步驟

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

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

返回構(gòu)造出來(lái)的結(jié)果List 結(jié)果。

對(duì)于關(guān)聯(lián)的結(jié)果查詢,如果是多對(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é)果的查詢,只需要查詢數(shù)據(jù)庫(kù)一次,然后對(duì)結(jié)果的整合和組裝全部放在了內(nèi)存中。

以上是通過(guò)查詢User表所有信息來(lái)演示了一對(duì)多和多對(duì)一的映射對(duì)象處理。希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Springboot下swagger-ui.html訪問不到的解決方案

    Springboot下swagger-ui.html訪問不到的解決方案

    這篇文章主要介紹了Springboot下swagger-ui.html訪問不到的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • 初識(shí)Java一些常見的數(shù)據(jù)類型

    初識(shí)Java一些常見的數(shù)據(jù)類型

    這篇文章主要介紹Java一些常見的數(shù)據(jù)類型,Java是一種優(yōu)秀的程序設(shè)計(jì)語(yǔ)言,它具有令人賞心悅目的語(yǔ)法和易于理解的語(yǔ)義,下面文章小編就來(lái)簡(jiǎn)單介紹為什么說(shuō)Java是最好的語(yǔ)言并且介紹它的各種常見類型,需要的朋友可以參考一下
    2021-10-10
  • java RMI詳細(xì)介紹及實(shí)例講解

    java RMI詳細(xì)介紹及實(shí)例講解

    這篇文章主要介紹了java RMI詳細(xì)介紹及實(shí)例講解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼

    Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼

    本篇文章主要介紹了Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼,非常具有實(shí)用價(jià)值,有興趣的可以了解一下。
    2017-03-03
  • RabbitMQ消息中間件示例詳解

    RabbitMQ消息中間件示例詳解

    這篇文章主要給大家介紹了關(guān)于RabbitMQ消息中間件的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一學(xué)習(xí)學(xué)習(xí)吧
    2018-12-12
  • Spring實(shí)戰(zhàn)之容器后處理器操作示例

    Spring實(shí)戰(zhàn)之容器后處理器操作示例

    這篇文章主要介紹了Spring實(shí)戰(zhàn)之容器后處理器操作,結(jié)合實(shí)例形式分析了spring容器后處理器配置、使用操作技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2019-12-12
  • Java中Retry方法的簡(jiǎn)單實(shí)現(xiàn)

    Java中Retry方法的簡(jiǎn)單實(shí)現(xiàn)

    這篇文章主要介紹了Java中Retry方法的簡(jiǎn)單實(shí)現(xiàn),Retry主要是利用Java的lambda表達(dá)式和線程接口實(shí)現(xiàn)有返回值和無(wú)返回值的重試,思考了下就寫了一個(gè)簡(jiǎn)易R(shí)etry功能分享出來(lái),需要的朋友可以參考下
    2024-01-01
  • java 排序算法之歸并排序

    java 排序算法之歸并排序

    本文主要講解了排序算法中的歸并排序,文中運(yùn)用大量的圖片和代碼講解的非常詳細(xì),感興趣的朋友可以學(xué)習(xí)一下這篇文章,相信可以幫助到你
    2021-09-09
  • Javaweb 鼠標(biāo)移入移出表格顏色變化的實(shí)現(xiàn)

    Javaweb 鼠標(biāo)移入移出表格顏色變化的實(shí)現(xiàn)

    這篇文章主要介紹了Javaweb 鼠標(biāo)移入移出表格顏色變化的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09
  • elasticsearch索引index之put?mapping的設(shè)置分析

    elasticsearch索引index之put?mapping的設(shè)置分析

    這篇文章主要為大家介紹了elasticsearch索引index之put?mapping的設(shè)置分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-04-04

最新評(píng)論