Mybatis【配置檔案】就是這麼簡單

Java3y發表於2019-02-25

配置檔案和對映檔案還有挺多的屬性我還沒有講的,現在就把它們一一補全

對映檔案

在mapper.xml檔案中配置很多的sql語句,執行每個sql語句時,封裝為MappedStatement物件,mapper.xml以statement為單位管理sql語句

Statement的實際位置就等於namespace+StatementId

佔位符

在Mybatis中,有兩種佔位符

  • #{}解析傳遞進來的引數資料
  • ${}對傳遞進來的引數原樣拼接在SQL中

主鍵生成策略

如果我們在Hibernate中,當我們插入資料的時候,我們是可以選擇是UUID策略的...

那麼在Mybatis是怎麼做的呢??

UUID


	<!-- mysql的uuid生成主鍵 -->
	<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
		<selectKey keyProperty="id" order="BEFORE" resultType="string">
			select uuid()
		</selectKey>
		
		INSERT INTO USER(id,username,birthday,sex,address) VALUES(#{id},#{username},#{birthday},#{sex},#{address})
	</insert> 
複製程式碼

主鍵返回

如果我們一般插入資料的話,如果我們想要知道剛剛插入的資料的主鍵是多少,我們可以通過以下的方式來獲取

需求:

  • user物件插入到資料庫後,新記錄的主鍵要通過user物件返回,通過user獲取主鍵值。

解決思路:

  • 通過LAST_INSERT_ID()獲取剛插入記錄的自增主鍵值,在insert語句執行後,執行select LAST_INSERT_ID()就可以獲取自增主鍵。

mysql:

	<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
		<selectKey keyProperty="id" order="AFTER" resultType="int">
			select LAST_INSERT_ID()
		</selectKey>
		INSERT INTO USER(username,birthday,sex,address) VALUES(#{username},#{birthday},#{sex},#{address})
	</insert>
複製程式碼

oracle:

實現思路:

  • 先查詢序列得到主鍵,將主鍵設定到user物件中,將user物件插入資料庫。

	<!-- oracle
	在執行insert之前執行select 序列.nextval() from dual取出序列最大值,將值設定到user物件 的id屬性
	 -->
	<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User">
		<selectKey keyProperty="id" order="BEFORE" resultType="int">
			select 序列.nextval() from dual
		</selectKey>
		
		INSERT INTO USER(id,username,birthday,sex,address) VALUES( 序列.nextval(),#{username},#{birthday},#{sex},#{address})
	</insert> 
複製程式碼

resultMap

有的時候,我們看別的對映檔案,可能看不到以下這麼一段程式碼:


	<resultMap id="userListResultMap" type="user" >
 		<!-- 列名
 		id_,username_,birthday_
 		id:要對映結果集的唯 一標識 ,稱為主鍵
 		column:結果集的列名
 		property:type指定的哪個屬性中
 		 -->
 		 <id column="id_" property="id"/>
 		 <!-- result就是普通列的對映配置 -->
 		 <result column="username_" property="username"/>
 		 <result column="birthday_" property="birthday"/>
 	
 	</resultMap>
複製程式碼

因為,如果我們的資料表的欄位和JavaBean的屬性名稱是相同時,我們就不用上面那段程式碼了。Mybatis會自動幫我們把返回的結果進行封裝成JavaBean

那當我們資料表的欄位和JavaBean的屬性名稱不是相同時,我們就需要使用resultMap,也就是上面那段程式碼

  • 當然了,在正常情況下列名和JavaBean的屬性名一般都是不同的,因此還是需要resultMap的。

resultMap和resultType區別

resultType :指定輸出結果的型別(pojo、簡單型別、hashmap..),將sql查詢結果對映為java物件 。

  • 使用resultType注意:sql查詢的列名要和resultType指定pojo的屬性名相同,指定相同 屬性方可對映成功,如果sql查詢的列名要和resultType指定pojo的屬性名全部不相同,list中無法建立pojo物件的。

resultMap:將sql查詢結果對映為java物件。

  • 如果sql查詢列名和最終要對映的pojo的屬性名不一致**,使用resultMap將列名和pojo的屬性名做一個對應關係 (列名和屬性名對映配置)**

使用resultMap


	<resultMap id="userListResultMap" type="user" >
 		<!-- 列名
 		id_,username_,birthday_
 		id:要對映結果集的唯 一標識 ,稱為主鍵
 		column:結果集的列名
 		property:type指定的哪個屬性中
 		 -->
 		 <id column="id_" property="id"/>
 		 <!-- result就是普通列的對映配置 -->
 		 <result column="username_" property="username"/>
 		 <result column="birthday_" property="birthday"/>
 	
 	</resultMap>
複製程式碼

這裡寫圖片描述

這裡寫圖片描述


resultType和resultMap用法總結

resultType:

  • 作用:

    • 將查詢結果按照sql列名pojo屬性名一致性對映到pojo中。
  • 場合:

    • 常見一些明細記錄的展示,將關聯查詢資訊全部展示在頁面時,此時可直接使用resultType將每一條記錄對映到pojo中,在前端頁面遍歷list(list中是pojo)即可。

resultMap:

  • 使用association和collection完成一對一和一對多高階對映。

association:

  • 作用:
    • 將關聯查詢資訊對映到一個pojo類中。
  • 場合:
    • 為了方便獲取關聯資訊可以使用association將關聯訂單對映為pojo,比如:查詢訂單及關聯使用者資訊。

collection:

  • 作用:
    • 將關聯查詢資訊對映到一個list集合中。
  • 場合:
    • 為了方便獲取關聯資訊可以使用collection將關聯資訊對映到list集合中,比如:查詢使用者許可權範圍模組和功能,可使用collection將模組和功能列表對映到list中。

Collection在前面好像並沒有用過,下面就看一下它的用法:

Order與OrderDetails關係



package cn.itcast.mybatis.po;

import java.io.Serializable;
import java.util.Date;
import java.util.List;

public class Orders implements Serializable {
    private Integer id;

    private Integer userId;

    private String number;

    private Date createtime;

    private String note;
    
    //關聯使用者資訊
    private User user;
    
    //訂單明細
    private List<Orderdetail> orderdetails;
   

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public Integer getUserId() {
        return userId;
    }

    public void setUserId(Integer userId) {
        this.userId = userId;
    }

    public String getNumber() {
        return number;
    }

    public void setNumber(String number) {
        this.number = number == null ? null : number.trim();
    }

    public Date getCreatetime() {
        return createtime;
    }

    public void setCreatetime(Date createtime) {
        this.createtime = createtime;
    }

    public String getNote() {
        return note;
    }

    public void setNote(String note) {
        this.note = note == null ? null : note.trim();
    }

	public User getUser() {
		return user;
	}

	public void setUser(User user) {
		this.user = user;
	}

	public List<Orderdetail> getOrderdetails() {
		return orderdetails;
	}

	public void setOrderdetails(List<Orderdetail> orderdetails) {
		this.orderdetails = orderdetails;
	} 
    
}

複製程式碼

SQL語句


	 <!-- 一對多查詢使用reusltMap完成
	查詢訂單關聯查詢訂單明細
	 -->
	 <select id="findOrderAndOrderDetails" resultMap="orderAndOrderDetails" >
			SELECT 
	  orders.*,
	  user.username,
	  user.sex ,
	  orderdetail.id orderdetail_id,
	  orderdetail.items_num,
	  orderdetail.items_id
	FROM
	  orders,
	  USER,
	  orderdetail
	WHERE orders.user_id = user.id  AND orders.id = orderdetail.orders_id
	 </select>
複製程式碼

resultMap


	<!-- 一對多,查詢訂單及訂單明細 -->
	<resultMap type="orders" id="orderAndOrderDetails" extends="ordersUserResultMap">
		<!-- 對映訂單資訊,和使用者資訊,這裡使用繼承ordersUserResultMap -->
		
		<!-- 對映訂單明細資訊 
		property:要將關聯資訊對映到orders的哪個屬性中
		ofType:集合中pojo的型別
		-->
		<collection property="orderdetails" ofType="cn.itcast.mybatis.po.Orderdetail">
			<!-- id:關聯資訊訂單明細的唯 一標識
			property:Orderdetail的屬性名
			  -->
			<id column="orderdetail_id" property="id"/>
			<result column="items_num" property="itemsNum"/>
			<result column="items_id" property="itemsId"/>
		</collection>
	
	</resultMap>
複製程式碼

一般地使用resultMap會多一點。


Mybatis對映檔案處理特殊字元

第一種方法:

  • 用了轉義字元把>和<替換掉,然後就沒有問題了。
  • SELECT * FROM test WHERE 1 = 1 AND start_date <= CURRENT_DATE AND end_date >= CURRENT_DATE

第二種方法:

  • <![CDATA[ ]]>

配置檔案

別名

typeAliases別名:

這裡寫圖片描述

自定義別名:


	<!-- 定義 別名 -->
	<typeAliases>
		<!--
		單個別名的定義
		alias:別名,type:別名對映的型別  -->
		<!-- <typeAlias type="cn.itcast.mybatis.po.User" alias="user"/> -->
		<!-- 批量別名定義
		指定包路徑,自動掃描包下邊的pojo,定義別名,別名預設為類名(首字母小寫或大寫)
		 -->
		<package name="cn.itcast.mybatis.po"/>
		
	</typeAliases>
複製程式碼

Mapper載入


	<mappers>
		<!-- 通過resource引用mapper的對映檔案 -->
		<mapper resource="sqlmap/User.xml" />
		<!-- <mapper resource="mapper/UserMapper.xml" /> -->
		<!-- 通過class引用mapper介面 
		class:配置mapper介面全限定名
		要求:需要mapper.xml和mapper.java同名並且在一個目錄 中
		-->
		<!-- <mapper class="cn.itcast.mybatis.mapper.UserMapper"/> -->
		<!-- 批量mapper配置 
		通過package進行自動掃描包下邊的mapper介面,
		要求:需要mapper.xml和mapper.java同名並且在一個目錄 中
		
		-->
		<package name="cn.itcast.mybatis.mapper"/>
	</mappers>
複製程式碼

延遲載入

在進行資料查詢時,為了提高資料庫查詢效能,儘量使用單表查詢,因為單表查詢比多表關聯查詢速度要快。

如果查詢單表就可以滿足需求,一開始先查詢單表,當需要關聯資訊時,再關聯查詢,當需要關聯資訊再查詢這個叫延遲載入

在Mybatis中延遲載入就是在resultMap中配置具體的延遲載入..

這裡寫圖片描述

在Mybatis的檔案中配置全域性延遲載入


	<!-- 全域性配置引數 -->
	<settings>
		<!-- 延遲載入總開關 -->
		<setting name="lazyLoadingEnabled" value="true" />	
		<!-- 設定按需載入 -->
		<setting name="aggressiveLazyLoading" value="false" />
	</settings>
複製程式碼

延遲載入測試

當需要使用者時呼叫 Orders類中的getUser()方法執行延遲載入 ,向資料庫發出sql。

由於是對User進行延遲載入,那麼我們只要查詢Orders相關的資訊即可了


	 <!-- 一對一查詢延遲載入
	 開始只查詢訂單,對使用者資訊進行延遲載入 
	  -->
	 <select id="findOrderUserListLazyLoading" resultMap="orderCustomLazyLoading">
		 SELECT 
		  orders.*
		FROM
		  orders
	 </select>
複製程式碼

使用resultMap來配置延遲載入


	<!-- 一對一查詢延遲載入 的配置 -->
	<resultMap type="orders" id="orderCustomLazyLoading">
		<!-- 完成了訂單資訊的對映配置 -->
		<!-- id:訂單關聯使用者查詢的唯 一 標識 -->
		<id column="id" property="id" />
		<result column="user_id" property="userId" />
		<result column="number" property="number" />
		<result column="createtime" property="createtime" />
		<result column="note" property="note" />
		<!--

		配置使用者資訊的延遲載入
		 	select:延遲載入執行的sql所在的statement的id,如果不在同一個namespace需要加namespace
			sql:根據使用者id查詢使用者資訊【column就是引數】
			column:關聯查詢的列
			property:將關聯查詢的使用者資訊設定到Orders的哪個屬性 -->


		<!--當需要user資料的時候,它就會把column所指定的user_id傳遞過去給cn.itcast.mybatis.mapper.UserMapper.findUserById作為引數來查詢資料-->
		<association property="user"
			select="cn.itcast.mybatis.mapper.UserMapper.findUserById" column="user_id"></association>

	</resultMap>
複製程式碼

這裡寫圖片描述

總結

  • 在程式中呼叫的SQL語句是由對映檔案的命令空間+sql片段的id所組成的。它內部會生成一個Statement物件的。
  • 在使用別名的時候,可以指定包名,在使用總配置檔案載入對映檔案時,也可以指定包名。
  • 主鍵如果需要返回的話,使用selectKey 標籤即可。UUID也可以返回。在Oracle的話,是使用序列來返回自動增長的主鍵的。
  • 佔位符有兩種,一種是解析傳遞進來的引數資料、一種是原樣輸出傳遞進來的資料。

如果文章有錯的地方歡迎指正,大家互相交流。習慣在微信看技術文章,想要獲取更多的Java資源的同學,可以關注微信公眾號:Java3y

相關文章