偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

聊聊Mybatis系列之Mapper接口

開發(fā) 開發(fā)工具
了解一個類,首先看下成員變量和構造方法。這里 config 不用多說了吧,主要的是 knownMappers 這個成員變量。這就是個map 對象,只是這個 map 對象的 value值是個對象,所以又要去看下 MapperProxyFactory 這個對象。

[[327761]]

 1.上期回顧

首先,我們還是回顧一下上篇文件的類容。先看下這個測試類,大家還有印象嗎:

  1. public class MybatisTest { 
  2.     @Test 
  3.     public void testSelect() throws IOException { 
  4.         String resource = "mybatis-config.xml"
  5.         InputStream inputStream = Resources.getResourceAsStream(resource); 
  6.         SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream); 
  7.         SqlSession session = sqlSessionFactory.openSession(); 
  8.         try { 
  9.             FruitMapper mapper = session.getMapper(FruitMapper.class); 
  10.             Fruit fruit = mapper.findById(1L); 
  11.             System.out.println(fruit); 
  12.         } finally { 
  13.             session.close(); 
  14.         } 
  15.     } 

上篇源碼分析講了 mybatis 一級緩存的實現(xiàn)原理。這次,我們來了解下 mybatis 接口的創(chuàng)建。

2. mapper接口的創(chuàng)建流程

2.1 SqlSession的getMapper()

首先,我們來看下 FruitMapper mapper = session.getMapper(FruitMapper.class); 這段代碼,意思很簡單,根據傳入的class 獲取這個對象的實例。這個流程有點復雜,阿粉帶著大家來跟下源碼:

首先還是ctrl + 左鍵點擊 getMapper 方法,然后會進入到 SqlSession 的 getMapper() 方法。然后之前阿粉也帶著大家了解了, SqlSession 的默認實現(xiàn)類是 DefaultSqlSession ,所以我們直接看下 getMapper() 在 DefaultSqlSession 里面的實現(xiàn):

  1. @Override 
  2. public <T> T getMapper(Class<T> type) { 
  3.     return configuration.getMapper(type, this); 

2.2 Configuration 的getMapper()

這里從 configuration 里面去獲取, configuration 是全局配置對象,也就是上下文。參數 this 是當前的SqlSession 對象,繼續(xù)跟進去看下:

  1. public <T> T getMapper(Class<T> type, SqlSession sqlSession) { 
  2.     return mapperRegistry.getMapper(type, sqlSession); 

2.3 MapperRegistry 的getMapper()

mapperRegistry 對象是干什么的呢?繼續(xù)點進去:

  1. public <T> T getMapper(Class<T> type, SqlSession sqlSession) { 
  2.     final MapperProxyFactory<T> mapperProxyFactory = (MapperProxyFactory<T>) knownMappers.get(type); 
  3.     if (mapperProxyFactory == null) { 
  4.         throw new BindingException("Type " + type + " is not known to the MapperRegistry."); 
  5.     } 
  6.     try { 
  7.         return mapperProxyFactory.newInstance(sqlSession); 
  8.     } catch (Exception e) { 
  9.         throw new BindingException("Error getting mapper instance. Cause: " + e, e); 
  10.     } 

這里就不好看懂了,需要先看下了解下 MapperRegistry 這個類,我們一步一步來,跟著阿粉的思路走:

  1. public class MapperRegistry { 
  2.  
  3.   private final Configuration config; 
  4.   private final Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<>(); 
  5.  
  6.   public MapperRegistry(Configuration config) { 
  7.     this.config = config; 
  8.   } 
  9.     ... 

了解一個類,首先看下成員變量和構造方法。這里 config 不用多說了吧,主要的是 knownMappers 這個成員變量。這就是個map 對象,只是這個 map 對象的 value值是個對象,所以又要去看下 MapperProxyFactory 這個對象,點進去:

  1. public class MapperProxyFactory<T> { 
  2.   private final Class<T> mapperInterface; 
  3.   private final Map<Method, MapperMethod> methodCache = new ConcurrentHashMap<>(); 
  4.  
  5.   public MapperProxyFactory(Class<T> mapperInterface) { 
  6.     this.mapperInterface = mapperInterface; 
  7.   } 
  8.     ... 

首先,單獨看下這個類名 MapperProxyFactory ,取名是很有學問的,好的名字讓你一下就知道是干啥的。所以一看 MapperProxyFactory ,首先就會聯(lián)想到工廠模式,工廠模式是干啥的?創(chuàng)建對象的,創(chuàng)建什么對象呢?創(chuàng)建 MapperProxy 對象的。MapperProxy 也是有玄機的,Proxy 的是什么?看到這個一般都是使用代理模式來創(chuàng)建代理對象的。所以就很清楚了, MapperProxyFactory 這個類就是個工廠,創(chuàng)建的是 mapper 的代理對象。

然后這個類里面存的是 mapper 的接口和接口里面的方法。

最后,我們回到 MapperRegistry 類里面的 getMapper() 方法?,F(xiàn)在是不是要清楚一些,通過 mapper 接口去 map 里面獲取工廠類 MapperProxyFactory ,然后通過工廠類去創(chuàng)建我們的 mapper 代理對象。然后在看下 getMapper() 方法里面的 mapperProxyFactory.newInstance(sqlSession); 這段代碼,繼續(xù)點進去:

  1. public T newInstance(SqlSession sqlSession) { 
  2.     final MapperProxy<T> mapperProxy = new MapperProxy<>(sqlSession, mapperInterface, methodCache); 
  3.     return newInstance(mapperProxy); 

你看,阿粉猜測對不對,MapperProxy 對象是不是出來了。然后看 newInstance() 這個方法:

  1. protected T newInstance(MapperProxy<T> mapperProxy) { 
  2.     return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy); 
  3.   } 

兩個 newInstance() 方法都在MapperProxyFactory 這個類里面,這里就很明顯嘛。典型的 JDK 代理對象的創(chuàng)建。

好了,到這里我們的 mapper對象就獲取到了。大家可以想一想,為什么獲取一個 mapper 對象會那么復雜?或者說 mapper 對象有什么作用?其實就是為了通過 mapper 接口的方法獲取到 mapper.xml 里面的 sql,具體怎么獲取的,請允許阿粉賣個關子,請聽阿粉下回分解。

3.總結

最后,阿粉以一個時序圖來結束本篇文章,喜歡的話,記得點個贊哦。么么噠~

 

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2022-10-20 18:00:00

MyBatis緩存類型

2022-06-02 07:11:13

JVMJava

2020-12-08 12:24:55

接口測試Interface

2021-01-14 10:00:57

Restful接口

2023-09-21 08:05:49

Mybatis插件開發(fā)

2021-12-09 12:22:28

MyBatis流程面試

2021-06-07 08:39:58

SpringBootMyBatisMapper

2023-07-26 07:13:55

函數接口Java 8

2022-12-28 08:16:16

metric聚合java

2025-01-16 10:30:49

2021-09-18 09:45:33

前端接口架構

2023-11-20 08:01:38

并發(fā)處理數Tomcat

2024-06-12 08:36:25

2021-07-11 12:12:49

.NETJWTjson

2022-02-08 23:59:12

USB接口串行

2025-01-07 09:07:36

接口屬性路徑

2022-01-26 00:05:00

接口Spring管理器

2022-11-17 07:43:13

2022-07-11 09:00:37

依賴配置文件Mybati

2021-06-08 09:28:12

.Net通知服務
點贊
收藏

51CTO技術棧公眾號