在MyBatis中,#{}和${}的区别是什么
#{}是预编译处理,${}是字符串替换
MyBatis在处理#{}时,会将SQL中的#{}替换为?号,调用PreparedStatement的set方法来赋值;
MyBatis在处理${}时,就是把${}替换成变量的值
使用#{}可以有效的防止SQL注入,提高系统安全性
阐述Mapper接口原理
一般的,我们在使用MyBatis操作数据时,只需要定义一个mapper xml,和一个与之对应的Mapper接口即可,但是并不需要提供Mapper接口的实现类,可以这么做的原因是MyBatis基于JDK的动态代理技术为我们的Mapper接口生成了动态代理类,操作的时候调用的也是该动态代理类对象的方法。
在解析全局配置文件过程中,有一个很重要的步骤就是解析mapper标签对应的mapper xml或者是Mapper接口注解类,在解析的过程中会使用接口信息创建MapperProxyFactory类,这是一个用于创建MapperProxy的工厂类,而MapperProxy是java.lang.reflect.InvocationHandler的实现类(用于使用动态代理创建Mapper接口实现类对象的),然后我们调用sqlSession.getMapper()方法时,内部就会调用MapperProxyFactory的newInstance方法获取对应的Mapper接口实现类对象,最后当我们调用Mapper接口实现类对象的任何方法时,就会去调用到这个MapperProxy类型对象的invoke方法,其里面本质还是以前命名空间的方式。
![](https://i-blog.csdnimg.cn/blog_migrate/7ce0af339181ccff43fcf682961c1da6.png)
MyBstis是否支持延迟加载?如果支持,它的实现原理是什么?
MyBatis仅支持association关联对象和collection关联集合对象的延迟加载,association指的就是一对一,collection指的就是一对多查询。在MyBatis配置文件中,可以配置是否启用延迟加载lazyLoadingEnabled=true|false。
它的原理是,使用Javasissit(默认)创建目标对象的代理对象,当调用目标方法时,进入拦截器方法,比如调用a.getB().getName(),拦截器invoke()发现a.getB()是null值,那么就会单独发送事先保存好的查询关联B对象的SQL,把B查询上来,然后调用a.setB(b),于是a的对象b属性就有值了,接着完成a.getB().getName()方法的调用。这就是延迟加载的基本原理。
MyBatis中如何执行批处理?
使用BatchExecutor完成批处理。执行update(没有select,JDBC批处理不支持select),将所有SQL都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个Statement对象,每个Statement对象都是addBatch()完毕后,等待逐一执行executeBatch()批处理。与JDBC批处理相同。