動態代理的功能:經過攔截器方法回調,對目標target方法進行加強。java
言外之意就是爲了加強目標target方法。上面這句話沒錯,但也不要認爲它就是真理,卻不知,動態代理還有投鞭斷流的霸權,連目標target都不要的科幻模式。程序員
注:本文默認認爲,讀者對動態代理的原理是理解的,若是不明白target的含義,難以看懂本篇文章,建議先理解動態代理。sql
首先定義一個pojo。apache
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 }123456789101112
再定義一個接口UserMapper.java。app
public interface UserMapper { public User getUserById(Integer id); }123
接下來咱們看看如何使用動態代理之投鞭斷流,實現實例化接口並調用接口方法返回數據的。
自定義一個InvocationHandler。ide
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); } }123456789101112131415161718192021222324
上面代碼中的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()); }123456789101112
output:測試
ID:1001 Name:zhangsan Age:18 x.y.MapperProxy@6bc7c0541234
這即是Mybatis自動映射器Mapper的底層實現原理。
可能有讀者不由要問:你怎麼把代碼寫的像初學者寫的同樣?沒有結構,且缺少美感。
必須聲明,做爲一名經驗老道的高手,能把程序寫的像初學者寫的同樣,那一定是高手中的高手。這樣可讓初學者感受到親切,舒服,符合本身的Style,讓他們或她們,感受到大牛寫的代碼也不過如此,本身甚至寫的比這些大牛寫的還要好,今後自信滿滿,熱情高漲,認爲與大牛之間的差距,僅剩下三分鐘。this
首先編寫一個測試類:spa
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(); } }123456789101112
Mapper長這個樣子:
public interface StudentMapper { List<Student> findAllStudents(); Student findStudentById(Integer id); void insertStudent(Student student); }12345
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); } // ...123456789101112131415161718192021222324252627
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); }12345678
這即是Mybatis使用動態代理之投鞭斷流。
相似下面:
public User getUserById(Integer id); public User getUserById(Integer id, String name);12
Answer:不能。
緣由:在投鞭斷流時,Mybatis使用package+Mapper+method全限名做爲key,去xml內尋找惟一sql來執行的。相似:key=x.y.UserMapper.getUserById,那麼,重載方法時將致使矛盾。對於Mapper接口,Mybatis禁止方法重載(overLoad)。
感謝你們看到這裏,若是本文有什麼不足之處,歡迎多多指教;若是你以爲對你有幫助,請給我點個贊。
也歡迎你們關注個人公衆號:程序員麥冬,麥冬天天都會分享java相關技術文章或行業資訊,歡迎你們關注和轉發文章!