答了Mybatis這個(gè)問題后,面試官叫我回去等通知……
背景
前段時(shí)間在我的技術(shù)群里,大家討論起了為什么UserMapper.java是個(gè)接口,沒有具體實(shí)現(xiàn)類,而我們可以直接調(diào)用其方法?
關(guān)于這個(gè)問題,我之前面試過一些人,很多人是這么回答的:
1.我領(lǐng)導(dǎo)叫我們使用Mybatis,大家都這么用就這么用了(沒想過,反正就這么用)。
2.雖然我不知道具體是怎么實(shí)現(xiàn)的,但我覺得肯定是……(此處略去若干的漫天猜想),但是也不對(duì)啊,難道是……(再次略去若干似懂非懂)。
3.使用動(dòng)態(tài)代理實(shí)現(xiàn)的(然后就沒有下文了)。
對(duì)于上面的三種回答,前面兩種我們就沒必要往下聊了。
但是第三種回答,就有必要往下問:那你說說動(dòng)態(tài)代理有哪些實(shí)現(xiàn)方式?Mybatis使用的是哪一種?
如果這個(gè)問題你還能回答上來,那么還會(huì)繼續(xù)問:UserMapper.java中大方法能不能重載?
如果你能回答上面的問題,本文就沒必要往下看了,已經(jīng)不適合你了。
問題分析
先來看一張圖,這圖里的代碼就是我們前面寫的demo:
為什么一個(gè)接口就能和一個(gè)xml文件給綁定的呢?這就是今天我們要聊的話題。
可能很多小伙伴不熟悉ibatis,2010年之前,還沒有Mybatis,之后ibatis便成了現(xiàn)在的Mybatis,如果有興趣的朋友,可以看到Mybatis中的包目錄。
這個(gè)包目錄中就還是ibatis,并且ibatis的作者現(xiàn)在就在騰訊上班,開發(fā)英雄聯(lián)盟LOL。
如果有騰訊的小伙伴可以打聽打聽哈,大佬就在身邊。言歸正傳。
Mapper層在Mybatis中現(xiàn)在是接口形式就搞定了,而在ibatis時(shí)代還是必須要有實(shí)現(xiàn)類的,我記得2012年的時(shí)候,使用的就是ibatis,Dao(Mapper)必須要有實(shí)現(xiàn)類。
下面我們就來看看Mybatis中是怎么做的。
使用案例
繼續(xù)使用我們上一節(jié)中的代碼。
controller
service實(shí)現(xiàn)類中
打一個(gè)斷點(diǎn),然后使用debug模式啟動(dòng)項(xiàng)目。并訪問:
http://localhost:9002/test
- userMapper=org.apache.ibatis.binding.MapperProxy@6da21078
發(fā)現(xiàn)Mybatis給UserMapper.java生成了一個(gè)代理對(duì)象,并且從名字上可以看出是JDK動(dòng)態(tài)代理。
關(guān)于動(dòng)態(tài)代理請(qǐng),這里我推薦我之前寫過的一篇文章:
https://gitbook.cn/m/mazi/activity/5d44e35e4fbf44126135c292?sut=c93c00a03b4f11eba07ad99b4dfbdab0&utm_source=chatweixinshare
其實(shí),又差不多回到了ibatis時(shí)代,只是Mybatis中是通過動(dòng)態(tài)代理的方式生成的代理類不是我們開發(fā)的,而是通過JDK動(dòng)態(tài)代理生成的代理類。
下面我們也使用JDK動(dòng)態(tài)代理來模擬一把。
- public class MapperProxy implements InvocationHandler {
- @SuppressWarnings("unchecked")
- public <T> T newInstance(Class<T> clz) {
- return (T) Proxy.newProxyInstance(clz.getClassLoader(), new Class[] { clz }, this);
- }
- @Override
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- if (Object.class.equals(method.getDeclaringClass())) {
- try {
- // 諸如hashCode()、toString()、equals()等方法,將target指向當(dāng)前對(duì)象this
- return method.invoke(this, args);
- } catch (Throwable t) {
- }
- }
- // 投鞭斷流
- return new User((Integer) args[0], "田維常", 22);
- }
- }
再寫一個(gè)測(cè)試類
- import com.tian.mybatis.entity.User;
- import com.tian.mybatis.mapper.UserMapper;
- public class TestProxy {
- public static void main(String[] args) {
- MapperProxy proxy = new MapperProxy();
- UserMapper mapper = proxy.newInstance(UserMapper.class);
- User user = mapper.selectById(999);
- System.out.println(user);
- System.out.println(mapper.toString());
- }
- }
輸出
- User{id=999, userName='田維常', age=22, gender=null}
- com.tian.mybatis.proxy.MapperProxy@39a054a5
這便是Mybatis自動(dòng)映射器Mapper的底層實(shí)現(xiàn)原理。
但是在Mybatis中,遠(yuǎn)遠(yuǎn)不是這么簡單的,但是本質(zhì)就是這樣的。
下面我們就來大致分析一下Mybatis中的這個(gè)流程。
接口Mapper內(nèi)的方法能重載嗎?
類似下面:
public User getUserById(Integer id);
public User getUserById(Integer id, String name);
答案:不能
因?yàn)镸ybatis中是使用package+Mapper+method全限名作為key,去xml內(nèi)尋找唯一sql來執(zhí)行的。
類似:key=com.tian.mybatis.UserMapper.getUserById,那么,重載方法時(shí)將導(dǎo)致矛盾。
對(duì)于Mapper接口,Mybatis禁止方法重載(overLoad) 。
在MapperMethod類的靜態(tài)內(nèi)部類中SqlCommand中有個(gè)resolveMappedStatement方法。
在Configuration中有個(gè)屬性,就是項(xiàng)目啟動(dòng)的時(shí)候,會(huì)把Mapper.xml中信息解析到這個(gè)屬性里,以我們指定的namespace+method作為key放到Map里面,后面我們調(diào)用Mapper接口動(dòng)態(tài)類的某個(gè)方法時(shí)候再去map獲取。
- protected final Map<String, MappedStatement> mappedStatements
就是使用類的全路徑名.方法作為key存放到Map中的。
總結(jié)
常用動(dòng)態(tài)代理方式:JDK動(dòng)態(tài)代理和CGlib動(dòng)態(tài)代理。
Mybatis是采用JDK動(dòng)態(tài)代理+反射+xml來解決接口綁定的,為我們創(chuàng)建可以調(diào)用的代理對(duì)象。
我們的Mapper中的方法是絕對(duì)不能重載的。
前端小智 本文轉(zhuǎn)載自微信公眾號(hào)「Java后端技術(shù)全棧」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系Java后端技術(shù)全棧公眾號(hào)。






































