前言很重要
動態代理的功能:通過攔截器方法回呼,對目標target方法進行增強,
言外之意就是為了增強目標target方法,上面這句話沒錯,但也不要認為它就是真理,殊不知,動態代理還有投鞭斷流的霸權,連目標target都不要的科幻模式,
注:本文默認認為,讀者對動態代理的原理是理解的,如果不明白target的含義,難以看懂本篇文章,建議先理解動態代理,
最新2020整理收集的一些面試題(都整理成檔案),有很多干貨,包含mysql,netty,spring,執行緒,spring cloud等詳細講解,也有詳細的學習規劃圖,面試題整理等,我感覺在面試這塊講的非常清楚:獲取面試資料只需:[點擊這里領取!!!] 暗號:CSDN
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
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),
注:學習時,是先研究的原始碼,看懂了原理,寫博文時,則先闡釋原理,再閱讀的原始碼,順序剛好相反,希望讀者不要因此疑惑,以為我強大到未卜先知,
最后:
最新2020整理收集的一些面試題(都整理成檔案),有很多干貨,包含mysql,netty,spring,執行緒,spring cloud等詳細講解,也有詳細的學習規劃圖,面試題整理等,我感覺在面試這塊講的非常清楚:獲取面試資料只需:點擊這里領取!!!暗號:CSDN
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/204107.html
標籤:java
上一篇:復盤逆襲之路!三個月時間深造,怒斬拼多多、位元組、螞蟻金服破50W年薪Offer(含自學路線圖)
下一篇:java基礎詳細講解
