一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Mybatis接口Mapper內的方法為啥不能重載嗎

Mybatis接口Mapper內的方法為啥不能重載嗎

2020-09-10 00:40程序員麥冬 Java教程

這篇文章主要介紹了Mybatis接口Mapper內的方法為啥不能重載嗎,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

動態代理的功能:通過攔截器方法回調,對目標target方法進行增強。

言外之意就是為了增強目標target方法。上面這句話沒錯,但也不要認為它就是真理,殊不知,動態代理還有投鞭斷流的霸權,連目標target都不要的科幻模式。

注:本文默認認為,讀者對動態代理的原理是理解的,如果不明白target的含義,難以看懂本篇文章,建議先理解動態代理。

1. 自定義JDK動態代理之投鞭斷流實現自動映射器Mapper

首先定義一個pojo。

?
1
2
3
4
5
6
7
8
9
10
11
12
public class User {
 private Integer id;
 private String name;
 private int age;
 
 public User(Integer id, String name, int age) {
  this.id = id;
  this.name = name;
  this.age = age;
 }
 // getter setter
}

再定義一個接口UserMapper.java。

?
1
2
3
public interface UserMapper {
 public User getUserById(Integer id);
}

接下來我們看看如何使用動態代理之投鞭斷流,實現實例化接口并調用接口方法返回數據的。

自定義一個InvocationHandler。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
 
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指向當前對象this
    return method.invoke(this, args);
   } catch (Throwable t) {
   }
  }
  // 投鞭斷流
  return new User((Integer) args[0], "zhangsan", 18);
 }
}

上面代碼中的target,在執行Object.java內的方法時,target被指向了this,target已經變成了傀儡、象征、占位符。在投鞭斷流式的攔截時,已經沒有了target。

寫一個測試代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
public static void main(String[] args) {
 MapperProxy proxy = new MapperProxy();
 
 UserMapper mapper = proxy.newInstance(UserMapper.class);
 User user = mapper.getUserById(1001);
 
 System.out.println("ID:" + user.getId());
 System.out.println("Name:" + user.getName());
 System.out.println("Age:" + user.getAge());
 
 System.out.println(mapper.toString());
}

output:

ID:1001
Name:zhangsan
Age:18
x.y.MapperProxy@6bc7c0541234

這便是Mybatis自動映射器Mapper的底層實現原理。

可能有讀者不禁要問:你怎么把代碼寫的像初學者寫的一樣?沒有結構,且缺乏美感。
必須聲明,作為一名經驗老道的高手,能把程序寫的像初學者寫的一樣,那必定是高手中的高手。這樣可以讓初學者感覺到親切,舒服,符合自己的Style,讓他們或她們,感覺到大牛寫的代碼也不過如此,自己甚至寫的比這些大牛寫的還要好,從此自信滿滿,熱情高漲,認為與大牛之間的差距,僅剩下三分鐘。

2. Mybatis自動映射器Mapper的源碼分析

首先編寫一個測試類:

?
1
2
3
4
5
6
7
8
9
10
11
12
public static void main(String[] args) {
  SqlSession sqlSession = MybatisSqlSessionFactory.openSession();
  try {
   StudentMapper studentMapper = sqlSession.getMapper(StudentMapper.class);
   List<Student> students = studentMapper.findAllStudents();
   for (Student student : students) {
    System.out.println(student);
   }
  } finally {
   sqlSession.close();
  }
 }

Mapper長這個樣子:

?
1
2
3
4
5
public interface StudentMapper {
 List<Student> findAllStudents();
 Student findStudentById(Integer id);
 void insertStudent(Student student);
}

org.apache.ibatis.binding.MapperProxy.java部分源碼。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class MapperProxy<T> implements InvocationHandler, Serializable {
 
 private static final long serialVersionUID = -6424540398559729838L;
 private final SqlSession sqlSession;
 private final Class<T> mapperInterface;
 private final Map<Method, MapperMethod> methodCache;
 
 public MapperProxy(SqlSession sqlSession, Class<T> mapperInterface, Map<Method, MapperMethod> methodCache) {
  this.sqlSession = sqlSession;
  this.mapperInterface = mapperInterface;
  this.methodCache = methodCache;
 }
 
 @Override
 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  if (Object.class.equals(method.getDeclaringClass())) {
   try {
    return method.invoke(this, args);
   } catch (Throwable t) {
    throw ExceptionUtil.unwrapThrowable(t);
   }
  }
  // 投鞭斷流
  final MapperMethod mapperMethod = cachedMapperMethod(method);
  return mapperMethod.execute(sqlSession, args);
 }
 // ...

org.apache.ibatis.binding.MapperProxyFactory.java部分源碼。

?
1
2
3
4
5
6
7
8
public class MapperProxyFactory<T> {
 
 private final Class<T> mapperInterface;
 
 @SuppressWarnings("unchecked")
 protected T newInstance(MapperProxy<T> mapperProxy) {
  return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
 }

這便是Mybatis使用動態代理之投鞭斷流。

3. 接口Mapper內的方法能重載(overLoad)嗎?(重要)

類似下面:

?
1
2
public User getUserById(Integer id);
public User getUserById(Integer id, String name);

Answer:不能。

原因:在投鞭斷流時,Mybatis使用package+Mapper+method全限名作為key,去xml內尋找唯一sql來執行的。類似:key=x.y.UserMapper.getUserById,那么,重載方法時將導致矛盾。對于Mapper接口,Mybatis禁止方法重載(overLoad)。

最后

到此這篇關于Mybatis接口Mapper內的方法為啥不能重載嗎的文章就介紹到這了,更多相關Mybatis接口Mapper重載內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://blog.51cto.com/14849432/2529253

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 操操久久| 好大好深受不了了快进来 | 22222色男人的天堂 | 精品欧美一区二区精品久久 | www在线免费观看 | 9420高清完整版在线观看国语 | 国产精品久久久天天影视香蕉 | 大乳一级一区二区三区 | 国产日韩欧美在线观看不卡 | 奇米色7777 | 99青青青精品视频在线 | 精品小视频在线观看 | 无码毛片内射白浆视频 | 成人在线观看一区 | 亚洲 制服 欧美 中文字幕 | 和日本免费不卡在线v | 国产精品短视频 | 69日本xxxhd | 被老外玩爽的中国美女视频 | 激情艳妇之性事高h | 青草影院在线观看 | 污污在线免费观看 | 国产专区亚洲欧美另类在线 | 国产精品久久久99 | 国产日韩欧美一区 | 午夜伦理电影在线观免费 | 国产成人亚洲综合91精品555 | k逼| 国产成人亚洲精品一区二区在线看 | 国产精品国产香蕉在线观看网 | 五月天精品视频在线观看 | 四虎最新免费观看网址 | 教室里老师好紧h | 无限资源在线观看完整版免费下载 | 99久久6er热免费精品 | 亚洲欧美视频在线播放 | 国产亚洲精品九九久在线观看 | 欧美三级一区二区 | 国内精品 大秀视频 日韩精品 | 久久精品麻豆国产天美传媒果冻 | 男人天堂bt |