#{}和${}
的区别是什么?
#{}和${}的区别是什么?
在Mybatis中,有两种占位符html
#{}
解析传递进来的参数数据#{}
是预编译处理,${}是字符串替换。当实体类中的属性名和表中的字段名不同 ,怎么办 ?
第1种: 经过在查询的sql语句中定义字段名的别名,让字段名的别名和实体类的属性名一致java
<select id=”selectorder” parametertype=”int” resultetype=”me.gacl.domain.order”> select order_id id, order_no orderno ,order_price price form orders where order_id=#{id}; </select>
第2种: 经过<resultMap>
来映射字段名和实体类属性名的一一对应的关系mysql
<select id="getOrder" parameterType="int" resultMap="orderresultmap"> select * from orders where order_id=#{id} </select> <resultMap type=”me.gacl.domain.order” id=”orderresultmap”> <!–用id属性来映射主键字段–> <id property=”id” column=”order_id”> <!–用result属性来映射非主键字段,property为实体类属性名,column为数据表中的属性–> <result property = “orderno” column =”order_no”/> <result property=”price” column=”order_price” /> </reslutMap>
我认为第二种方式会好一点。程序员
如何获取自动生成的(主)键值?
若是咱们通常插入数据的话,若是咱们想要知道刚刚插入的数据的主键是多少,咱们能够经过如下的方式来获取面试
需求:sql
解决思路:数据库
mysql:缓存
<insert id="insertUser" parameterType="cn.itcast.mybatis.po.User"> <selectKey keyProperty="id" order="AFTER" resultType="int"> select LAST_INSERT_ID() </selectKey> INSERT INTO USER(username,birthday,sex,address) VALUES(#{username},#{birthday},#{sex},#{address}) </insert>
oracle:安全
实现思路:微信
<!-- oracle 在执行insert以前执行select 序列.nextval() from dual取出序列最大值,将值设置到user对象 的id属性 --> <insert id="insertUser" parameterType="cn.itcast.mybatis.po.User"> <selectKey keyProperty="id" order="BEFORE" resultType="int"> select 序列.nextval() from dual </selectKey> INSERT INTO USER(id,username,birthday,sex,address) VALUES( 序列.nextval(),#{username},#{birthday},#{sex},#{address}) </insert>
在mapper中如何传递多个参数?
第一种:使用占位符的思想
在映射文件中使用#{0},#{1}表明传递进来的第几个参数
@param
注解:来命名参数 #{0}
,#{1}方式//对应的xml,#{0}表明接收的是dao层中的第一个参数,#{1}表明dao层中第二参数,更多参数一致日后加便可。 <select id="selectUser"resultMap="BaseResultMap"> select * fromuser_user_t whereuser_name = #{0} anduser_area=#{1} </select>
public interface usermapper { user selectuser(@param(“username”) string username, @param(“hashedpassword”) string hashedpassword); }
<select id=”selectuser” resulttype=”user”> select id, username, hashedpassword from some_table where username = #{username} and hashedpassword = #{hashedpassword} </select>
第二种:使用Map集合做为参数来装载
try{ //映射文件的命名空间.SQL片断的ID,就能够调用对应的映射文件中的SQL /** * 因为咱们的参数超过了两个,而方法中只有一个Object参数收集 * 所以咱们使用Map集合来装载咱们的参数 */ Map<String, Object> map = new HashMap(); map.put("start", start); map.put("end", end); return sqlSession.selectList("StudentID.pagination", map); }catch(Exception e){ e.printStackTrace(); sqlSession.rollback(); throw e; }finally{ MybatisUtil.closeSqlSession(); }
<!--分页查询--> <select id="pagination" parameterType="map" resultMap="studentMap"> /*根据key自动找到对应Map集合的value*/ select * from students limit #{start},#{end}; </select>
Mybatis动态sql是作什么的?都有哪些动态sql?能简述一下动态sql的执行原理不?
详情Demo可参考我别的文章:
Mybatis的Xml映射文件中,不一样的Xml映射文件,id是否能够重复?
若是配置了namespace那么固然是能够重复的,由于咱们的Statement实际上就是namespace+id
若是没有配置namespace的话,那么相同的id就会致使覆盖了。
为何说Mybatis是半自动ORM映射工具?它与全自动的区别在哪里?
一般一个Xml映射文件,都会写一个Dao接口与之对应,请问,这个Dao接口的工做原理是什么?Dao接口里的方法,参数不一样时,方法能重载吗?
举例:
com.mybatis3.mappers.StudentDao.findStudentById, 能够惟一找到namespace为com.mybatis3.mappers.StudentDao下面id = findStudentById的MappedStatement。在Mybatis中,每个<select>、<insert>、<update>、<delete>标签,都会被解析为一个MappedStatement对象。
Dao接口里的方法,是不能重载的,由于是全限名+方法名的保存和寻找策略。
Dao接口的工做原理是JDK动态代理,Mybatis运行时会使用JDK动态代理为Dao接口生成代理proxy对象,代理对象proxy会拦截接口方法,转而执行MappedStatement所表明的sql,而后将sql执行结果返回。
详情可参考:
Mybatis比IBatis比较大的几个改进是什么
接口绑定有几种实现方式,分别是怎么实现的?
接口绑定有两种实现方式:
Mybatis是如何进行分页的?分页插件的原理是什么?
Mybatis使用RowBounds对象进行分页,它是针对ResultSet结果集执行的内存分页,而非物理分页,能够在sql内直接书写带有物理分页的参数来完成物理分页功能,也可使用分页插件来完成物理分页。
分页插件的基本原理是使用Mybatis提供的插件接口,实现自定义插件,在插件的拦截方法内拦截待执行的sql,而后重写sql,根据dialect方言,添加对应的物理分页语句和物理分页参数。
举例:select * from student,拦截sql后重写为:select t.* from (select * from student)t limit 0,10
分页插件使用参考资料:
简述Mybatis的插件运行原理,以及如何编写一个插件
Mybatis仅能够编写针对ParameterHandler、ResultSetHandler、StatementHandler、Executor这4种接口的插件,Mybatis使用JDK的动态代理,为须要拦截的接口生成代理对象以实现接口方法拦截功能,每当执行这4种接口对象的方法时,就会进入拦截方法,具体就是InvocationHandler的invoke()方法,固然,只会拦截那些你指定须要拦截的方法。
实现Mybatis的Interceptor接口并复写intercept()方法,而后在给插件编写注解,指定要拦截哪个接口的哪些方法便可,记住,别忘了在配置文件中配置你编写的插件。
Mybatis是否支持延迟加载?若是支持,它的实现原理是什么?
Mybatis仅支持association关联对象和collection关联集合对象的延迟加载,association指的就是一对一,collection指的就是一对多查询。在Mybatis配置文件中,能够配置是否启用延迟加载lazyLoadingEnabled=true|false。
它的原理是,使用CGLIB建立目标对象的代理对象,当调用目标方法时,进入拦截器方法,好比调用a.getB().getName(),拦截器invoke()方法发现a.getB()是null值,那么就会单独发送事先保存好的查询关联B对象的sql,把B查询上来,而后调用a.setB(b),因而a的对象b属性就有值了,接着完成a.getB().getName()方法的调用。这就是延迟加载的基本原理。
固然了,不光是Mybatis,几乎全部的包括Hibernate,支持延迟加载的原理都是同样的。
Mybatis都有哪些Executor执行器?它们之间的区别是什么?
Mybatis有三种基本的Executor执行器,SimpleExecutor、ReuseExecutor、BatchExecutor。
做用范围:Executor的这些特色,都严格限制在SqlSession生命周期范围内。
MyBatis与Hibernate有哪些不一样?
Mybatis和hibernate不一样,它不彻底是一个ORM框架,由于MyBatis须要程序员本身编写Sql语句,不过mybatis能够经过XML或注解方式灵活配置要运行的sql语句,并将java对象和sql语句映射生成最终执行的sql,最后将sql执行的结果再映射生成java对象。
Mybatis学习门槛低,简单易学,程序员直接编写原生态sql,可严格控制sql执行性能,灵活度高,很是适合对关系数据模型要求不高的软件开发,例如互联网软件、企业运营类软件等,由于这类软件需求变化频繁,一但需求变化要求成果输出迅速。可是灵活的前提是mybatis没法作到数据库无关性,若是须要实现支持多种数据库的软件则须要自定义多套sql映射文件,工做量大。
Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)若是用hibernate开发能够节省不少代码,提升效率。可是Hibernate的缺点是学习门槛高,要精通门槛更高,并且怎么设计O/R映射,在性能和对象模型之间如何权衡,以及怎样用好Hibernate须要具备很强的经验和能力才行。
总之,按照用户的需求在有限的资源环境下只要能作出维护性、扩展性良好的软件架构都是好架构,因此框架只有适合才是最好。
参考资料:
若是文章有错的地方欢迎指正,你们互相交流。 习惯在微信看技术文章,想要获取更多的Java资源的同窗,能够关注微信公众号:Java3y