MyBatis之Mapper動態代理開發

1.SqlSession的使用範圍

1.SqlSessionFactoryBuilder
SqlSessionFactoryBuilder是以工具類的方式來使用:須要建立sqlSessionFactory時就new一個 SqlSessionFactoryBuilderjava

2.sqlSessionFactory
正常開發時,以單例方式管理sqlSessionFactory,整個系統運行過程當中sqlSessionFactory只有一個實例,未來和Spring整合後由Spring以單例方式管理sqlSessionFactory程序員

3.SqlSession
SqlSession是一個面向用戶(程序員)的接口,程序員調用 SqlSession接口的方法進行操做數據庫。那麼咱們會思考:SqlSession可否以單例方式使用???因爲 SqlSession是線程不安全的,因此 SqlSession最佳應用範圍在方法體內。也就是說在方法體內定義局部變量 SqlSession的對象來使用。sql

2.MyBatis開發DAO的方式

咱們先來看看MyBatis原始開發dao的開發方式,發現原始開發的問題,而後再來看看MyBatis使用mapper動態代理開發dao的方式(也是MyBatis目前使用的開發dao的方式)。數據庫

2.1原始dao的開發方式

程序員須要編寫dao接口:和dao接口的實現類:安全

而後就能在測試類中使用。測試類代碼以下:mybatis

咱們來看看這種方式開發有什麼問題?架構

  • 1.dao的實現類中存在重複代碼,整個mybatis操做的過程代碼模板重複(都是先建立sqlSession、調用sqlSession的方法、關閉sqlSession)。
  • 2.dao的實現類中存在硬編碼,調用sqlSession方法時將statement的id硬編碼。

下面咱們看看mapper動態代理的方式。app

2.2mapper動態代理的方式

這種方式下程序員只須要寫dao接口,dao接口實現對象由mybatis自動生成代理對象。由於自己dao在三層架構中就是一個通用的接口。工具

2.2.1mapper開發規範

要想讓mybatis自動建立dao接口實現類的代理對象,必需要遵循一些規則:測試

  • 1.mapper.xml中 namespace指定爲mapper接口的全限定名。此步驟的目的:將mapper.xml和mapper.java關聯。
  • 2.mapper.xml中statement的id就是mapper.java中的方法名。
  • 3.mapper.xml中statement的parameterType和mapper.java中方法輸入參數一致。
  • 4.mapper.xml中statement的resultType和mapper.java中方法的返回值類型一致。

採用這種方式後,咱們即可將第一篇文章中提到的User.xml改成UserMapper.xml。文件目錄以下:其中有些類咱們會在後面用到。

2.2.2mapper.xml(映射文件)

mapper映射文件的命名方式建議表名加Mapper.xml,namespace指定爲mapper接口的全限定名。

2.2.3mapper.java接口

mybatis提出了mapper接口,至關於dao接口,mapper接口的命名方式建議爲表名加Mapper.

1
public interface UserMapper{};

 

2.3.4將mapper.xml在SqlMapConfing.xml中進行註冊

<mappers>
 <mapper resource="mapper/UserMapper.xml"/>
</mappers>

2.3.5mapper接口返回單個對象和集合對象

在UserMapper.java中添加以下兩個方法:

對於UserMapper.xml,無論查詢記錄是單條仍是多條,在statement(即UserMapper.xml)中的resultType都定義一致,都是單條記錄映射的pojo類型。

而對於UserMapper.java接口方法中的返回值,若是返回的是單個對象,返回值類型是pojo,生成的代理對象內部會自動經過selectOne獲取記錄,若是返回值類型是多條對象,生成的代理對象內部會自動經過selectList獲取記錄。

測試代碼以下:

使用Mapper代理方式進行開發,使程序員只須要關注UserMapper.java接口中的方法,它的實現類由Mapper自動爲咱們生成,帶來了很大的方便。但這種方式也有它的弊端。

2.3.6mapper代理開發的問題

  • 1.返回值的問題:若是方法(即UserMapper.java接口中的方法)調用的statement中返回是多條記錄,而mapper.java方法的返回值爲pojo,此時代理對象經過selectOne調用,但因爲返回的是多條記錄因此會報錯:Expected one result (for null ) to be returned by selectOne() but found 4;
  • 2.輸入參數的問題:使用mapper代理的方式開發,mapper接口方法的輸入參數只有一個,可擴展性是否不好?答:可擴展性沒有問題,由於dao層就是通用的,能夠經過擴展pojo(定義pojo包裝類型,後面第四篇文章–MyBatis輸入輸出映射會講擴展pojo的知識)來將不一樣的參數(能夠是pojo也能夠是簡單類型)傳入進去。
相關文章
相關標籤/搜索