Mybatis面试问题

Mybatis面试问题

一、JDBC编程有哪些不足的地方,Mybatis是如何解决这些问题的?

  1. 数据库连接的创建和销毁会造成系统资源浪费从而影响性能,如何使用数据库连接池来解决这个问题。

    **解决:**在SqlMapConfig.xml中配置数据库连接池,使用连接池管理数据库连接。

  2. Sql语句在编写时写在代码中造成代码不宜维护,实际使用的sql语句变化可能比较大,sql改动需要重新改写java代码。

    解决:将Sql语句配置在XXXXMapper.xml文件中,实现于java代码分离。

  3. 向sql语句中传入参数较为困难,因为sql语句的where不一定,可能多可能少,占位符要和参数一一对应。

    解决:Mybatis使用java对象映射到sql语句中。

  4. 对结果集解析麻烦,sql变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成pojo对象解析比较方便。

    解决:Mybatis自动将sql执行结果映射至java对象。

二、Mybatis编程步骤是什么

  1. 创建SqlSessionFactory
  2. 通过SqlSessionFactory创建SqlSession
  3. 通过SqlSession执行数据库操作
  4. 调用session.commit()提交事务
  5. 调用session.close()关闭事务

三、Mybatis与Hibernate有什么不同

​ Mybatis并不是完全的ORM框架,因为Mybatis需要程序员自己编写对应的Sql语句,不过mybatis可以通过xml或者注解的当时灵活的配置要运行的sql语句,并将java对象和sql语句进行映射生成最终的sql语句,对sql语句返回的结果最终生成对应的java对象。

​ Mybatis学习门槛低,简单易学,直接编写原生的sql,可严格控制sql 的性能,灵活度高,非常适合对关系模型数据要求不高的软件开发,例如互联网软件、企业运行类软件等,这些软件与需求无关,一旦需求变化可以迅速更改数据库。但是灵活性的前提是mybatis无法做到数据库无关性,如果需要支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。

​ Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果使用hibernate开发可以节省很多代码,提高效率。但是Hibernate缺点是学习门槛高,精通门槛高,而且怎么O/R映射,在性能和对象模型之间平衡,以及怎么用好Hibernate都需要很高的经验和能力。

四、使用MyBatis的mapper接口调用时有哪些要求?

  1. Mapper接口方法名和mapper.xml中定义的每个sql的id相同
  2. Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql的parameterType类型相同
  3. Mapper接口方法中输出的参数类型和mapper.xml中定义的每个sql的resultType相同
  4. Mapper.xml文件中的namespace即是Mapper接口的路径

五、SqlMapConfig.xml中配置有哪些内容?

SqlMapConfig.xml中配置的内容和顺序如下:

  • properties(属性)
  • setting(配置)
  • typeAliases(类型别名)
  • typeHandlers(类型处理器)
  • objectFactory(对象工厂)
  • plugins(插件)
  • environments(环境集合属性对象)
  • environment(环境子属性对象)
  • transactionManager(事务管理)
  • dataSource(数据源)
  • mapper(映射器)

六、简单的说一下MyBatis的一二级缓存?

MyBatis首先去缓存中查询结果集,如果没有则查询数据库,如果有从缓存中取出数据集,MyBatis内部缓存存储使用HashMap,key存放在CacheKey中使用其中的hashcode

  private static final int DEFAULT_HASHCODE = 17;

默认hashCode为17

多个对象时,对每个对象计算HashCode,在进行混淆

    int baseHashCode = object == null ? 1 : ArrayUtil.hashCode(object); 

    count++;
    checksum += baseHashCode;
    baseHashCode *= count;

    hashcode = multiplier * hashcode + baseHashCode;

    updateList.add(object);

SqlSession:该类是非线程安全的,其实例是不能共享的,所以应该每个线程具有自己的SqlSession实例。因此最佳建议是“请求或方法范围”。例如:收到一个Http请求后,创建一个实例,进行某些操作,之后关闭。确保将关闭放于finally中。

Mybatis的二级缓存即查询缓存,它的作用域是一个mapper的namespace,即在同一个namespace中查询sql可以从缓存中获取数据。二级缓存是可以跨SqlSession的,通过开启二级缓存,通过关联两个不同的命名空间,使用二级缓存属性类需要实现Serializable序列化接 口(可用来保存对象的状态)。

七、#{}${}有什么区别

  • #{}解析传递过来的参数数据

  • ${}对传递进来的参数拼接到SQL中

  • #{}是预编译的,${}是字符串替换。

  • 使用#{}可以有效防止SQL注入,提高系统安全性

  • 需要动态拼接表名时。

    SELECT * FROM ${tablename}
    
  • 动态拼接排序字段。

    SELECT * FROM TABLE ORDER BY ${username} 
    

八、如何获取自动生成的(主)键值?

需求:user对象在插入数据库后,新记录的主键要通过user对象返回,通过user获取主键值。

解决思路:

​ 通过LAST_INSERT_ID()获取刚插入的自增主键值,在insert语句执行后,执行select LAST_INSERT_ID()就可以获取自增主键。

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:

实现思路:

​ 先查询序列获得主键,将主键设置到user对象中,将user对象插入到数据库中。

 <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> 

Mybatis动态sql是做什么的?

  • Mybatis动态sql可以让我们在xml映射文件内,以标签的形式编写动态的sql,完成逻辑判断和动态拼接sql 的功能。

  • Mybatis提供9中动态sql标签:

    trim|where|set|foreach|if|choose|when|otherwise|bind。

  • 执行原理,使用OGNL从sql参数对象中计算表达式的值,根据表达式的值动态拼接sql,以此来完成动态sql的功能。

九、Mybatis的方法重载

Mybatis同一个mapper下面定义的相同的方法名,不同入参的函数,是不能进行重载的。因为使用的HashMap的Key为packageName+methodName。重名时会出现错误。

十、Mybatis基本结构

十一、参考

作者:沉沦2014

链接:https://www.jianshu.com/p/b7c591494fe8

來源:简书

简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

你可能感兴趣的:(Mybatis,Mybatis,面试,Java)