av激情亚洲男人的天堂国语,日韩欧美精品一中文字幕,无码av一区二区三区无码,国产又色又爽又刺激的a片,国产又色又爽又刺激的a片

聊聊Mybatis系列之Mapper接口

 1.上期回顧

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

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

 
 
 
 
  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 一級緩存的實(shí)現(xiàn)原理。這次,我們來了解下 mybatis 接口的創(chuàng)建。

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

2.1 SqlSession的getMapper()

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

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

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

2.2 Configuration 的getMapper()

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

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

2.3 MapperRegistry 的getMapper()

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

 
 
 
 
  1. public  T getMapper(Class type, SqlSession sqlSession) { 
  2.     final MapperProxyFactory mapperProxyFactory = (MapperProxyFactory) 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 這個(gè)類,我們一步一步來,跟著阿粉的思路走:

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

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

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

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

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

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

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

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

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

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

好了,到這里我們的 mapper對象就獲取到了。大家可以想一想,為什么獲取一個(gè) mapper 對象會(huì)那么復(fù)雜?或者說 mapper 對象有什么作用?其實(shí)就是為了通過 mapper 接口的方法獲取到 mapper.xml 里面的 sql,具體怎么獲取的,請?jiān)试S阿粉賣個(gè)關(guān)子,請聽阿粉下回分解。

3.總結(jié)

最后,阿粉以一個(gè)時(shí)序圖來結(jié)束本篇文章,喜歡的話,記得點(diǎn)個(gè)贊哦。么么噠~


分享標(biāo)題:聊聊Mybatis系列之Mapper接口
URL鏈接:http://uogjgqi.cn/article/dhhhcpe.html
掃二維碼與項(xiàng)目經(jīng)理溝通

我們在微信上24小時(shí)期待你的聲音

解答本文疑問/技術(shù)咨詢/運(yùn)營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流