@mapkey("id")指定返回类型为map时,将id作为key
我们写一个usermapper接口,mybatis会为改接口创建一个mapperproxy对象。
private final map<class<?>, mapperproxyfactory<?>> knownmappers = new hashmap<class<?>, mapperproxyfactory<?>>();
map以mapper接口类为key,value是接口类对应的代理工厂。void addmapper(class<t> type)
方法简单的将type放进knownmappers里面。不过,这里面还有mapperannotationbuilder parser = new mapperannotationbuilder(config, type); parser.parse();
这样一个解析操作,暂不做分析。
<t> t getmapper(class<t> type, sqlsession sqlsession)
该方法是本类最重要的一个方法了,获取mapper接口的代理对象也正是从该方法获取的。mapperproxyfactory.newinstance(sqlsession)
。public t newinstance(sqlsession sqlsession) { final mapperproxy<t> mapperproxy = new mapperproxy<t>(sqlsession, mapperinterface, methodcache); return newinstance(mapperproxy); }
再进去看
protected t newinstance(mapperproxy<t> mapperproxy) { return (t) proxy.newproxyinstance(mapperinterface.getclassloader(), new class[] { mapperinterface }, mapperproxy); }
发现这就是标准的jdk面向接口的动态代理啊。要问对应的invocationhandler在哪里?mapperproxy就是啦。
至此,mybatis如何获取mapper接口的代理对象已经解决。
再梳理一下整个流程:
sqlsession.getmapper->configuration.getmapper->mapperregistry.getmapper->获取mapper代理对象
创建好mapperproxy对象之后,调用mapper.xx方法时就会走mapperproxy的invoke方法
public object invoke(object proxy, method method, object[] args) throws throwable { try { if (object.class.equals(method.getdeclaringclass())) {//如果方法是在object里面的,放行 return method.invoke(this, args); } else if (isdefaultmethod(method)) { return invokedefaultmethod(proxy, method, args); } } catch (throwable t) { throw exceptionutil.unwrapthrowable(t); } final mappermethod mappermethod = cachedmappermethod(method);//缓存method方法 return mappermethod.execute(sqlsession, args); // 执行 }
发现关键点在于mappermethod.execute(sqlsession, args)
方法
如对本文有疑问, 点击进行留言回复!!
springmvc之ResponseBody响应json数据遇到的错误及解决
uni-app 后台升级 静默升级 uniapp 后台更新 静默更新 在线升级
SpringBoot多Module启动报错Could not transfer metadata
Hibernate项目报错:Cannot call sendError() after the response has been committed
网友评论