中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

mapper方法在Mybatis中為什么不能重載

發布時間:2020-12-25 14:30:36 來源:億速云 閱讀:136 作者:Leah 欄目:開發技術

mapper方法在Mybatis中為什么不能重載?針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

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

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

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

首先定義一個pojo。

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。

public interface UserMapper {
 public User getUserById(Integer id); 
}

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

自定義一個InvocationHandler。

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。

寫一個測試代碼:

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 page
x.y.MapperProxy@6bc7c054

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

可能有讀者不禁要問:你怎么把代碼寫的像初學者寫的一樣?沒有結構,且缺乏美感。

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

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

首先編寫一個測試類:

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長這個樣子:

public interface StudentMapper {
 List<Student> findAllStudents();
 Student findStudentById(Integer id);
 void insertStudent(Student student);
}

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

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部分源碼。

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)嗎?(重要)

類似下面:

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)。

關于mapper方法在Mybatis中為什么不能重載問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

平果县| 浮山县| 博野县| 平山县| 苍山县| 靖边县| 托克逊县| 镇沅| 喀喇| 岚皋县| 湖北省| 项城市| 安图县| 泽库县| 会泽县| 营口市| 嵩明县| 吉木萨尔县| 阜南县| 长沙县| 黑水县| 大姚县| 德州市| 济阳县| 承德县| 海安县| 淮北市| 肇庆市| 霍林郭勒市| 崇信县| 梅州市| 兰西县| 怀柔区| 西昌市| 长乐市| 焉耆| 福鼎市| 鄂伦春自治旗| 临江市| 广汉市| 赞皇县|