欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 房产 > 家装 > Mybatis

Mybatis

2025/6/16 9:34:51 来源:https://blog.csdn.net/2401_83176717/article/details/148583886  浏览:    关键词:Mybatis

resultType和resultMap的区别:

resultType

  • 简单映射:直接返回对应的Java类型(如基本类型,实体类或Map),要求数据库字段名和属性名完全匹配(或通过查询别名匹配)
  • 驼峰支持:开启驼峰映射后,Mybatis会自动将列名中的下划线去除并更改为驼峰命名形式
  • 自动映射:Mybatis会自动根据列名填充对象属性,无需额外配置
  • 限制:无法处理字段名与属性名不匹配的情况,不支持嵌套对象(对象内部包含其他对象)、一对多嵌套(对象内部包含其他对象的集合)等复杂关联映射
<select id="getUser" resultType="User">SELECT user_name FROM user  <!-- 自动映射到 userName -->
</select>

resultMap

  • 复杂映射:需要显式定义字段与属性的映射关系,支持以下功能:
  1. 字段名与属性名不一致:通过<result property="属性名" column="列名"/>配置
  2. 嵌套关系:使用<association property="表名" javaType="类名">处理一对一嵌套关系<collection>处理一对多嵌套关系
  3. 联表查询:可映射联合查询结果到多层级嵌套实体类
<resultMap id="userWithDeptMap" type="User"><id property="id" column="user_id"/><result property="name" column="user_name"/><!-- 嵌套映射部门对象 --><association property="dept" javaType="Dept"><result property="name" column="dept_name"/></association>
</resultMap>
  • 灵活性:适用于复杂业务场景,如连表查询、动态字段等。
Mybatis原生注解支持字段映射和嵌套关系
@Results({@Result(property = "orders", column = "user_id", many = @Many(select = "selectOrdersByUserId"))
})
@Select("SELECT * FROM user WHERE id = #{id}")
User getUserWithOrders(Long id);
@Results({@Result(property = "userName", column = "user_name"), // 字段别名映射@Result(property = "age", column = "age")
})
@Select("SELECT user_name, age FROM user WHERE id = #{id}")
User getUserById(int id);
MybatisPlus基于单表的增强式注解(不支持嵌套关系):

@TableName:指定表名和实体类的映射解决类名和表名不一致的问题

@TableField:指定字段名和属性名的映射支持自动驼峰映射

一级缓存和二级缓存:

对比项SqlSession(会话)​Connection(连接)​
定义MyBatis 的核心接口,表示一次数据库交互的上下文JDBC 的物理连接,直接与数据库通信
生命周期由 SqlSessionFactory 创建,手动或框架关闭从连接池获取,用完后归还(如 Druid、Hikari)
作用范围一个业务逻辑单元(如 HTTP 请求)单个 SQL 执行周期
线程安全非线程安全​(需每个线程独立实例)通常非线程安全(需避免多线程共享)
关联关系一个 SqlSession 可包含多个 Connection一个 Connection 仅属于一个 SqlSession

三者协作流程

  1. 请求进入​:业务代码通过 SqlSession 获取 Mapper 代理实例(如 UserMapper)。
  2. SQL 执行​:Mapper 方法触发时,SqlSession 从连接池获取 Connection 执行 SQL。
  3. 缓存与事务​:
    • 开启二级缓存后所有Mapper代理实例共享缓存
    • 事务由 SqlSession 控制(提交/回滚后释放 Connection)。

一级缓存

  • 作用域SqlSession会话级别(同一个业务会话内有效)
  • 默认开启:无需配置,自动开启
  • 存储位置:SqlSession对象内部
  • 缓存策略键值存储基于SQL+参数+分页条件作为Key缓存结果
  • 生效条件同一个SqlSession中查询完全一样的Sql语句时命中缓存
SqlSession session = sqlSessionFactory.openSession();
UserMapper mapper = session.getMapper(UserMapper.class);User user1 = mapper.selectById(1);  // 第一次查询数据库
User user2 = mapper.selectById(1);  // 从一级缓存获取session.close();  // 缓存失效
  • 失效条件
  1. 事务执行了增删改操作(无论是否提交)
  2. 手动清理了session缓存或关闭session
  3. 事务提交或回滚
  4. 跨SqlSession查询

二级缓存

  • 作用域:Mapper级别(跨SqlSession)
  • 需手动开启:XML或注解配置
<mapper namespace="com.example.UserMapper"><cache/>  <!-- 启用二级缓存 -->
</mapper>
@CacheNamespace
public interface UserMapper { ... }
  • 存储位置:内存或Redis缓存中
  • 缓存策略序列化存储,需要实体类实现Serializable接口启用序列化
  • 生效场景:跨SqlSession查询(多用户查询同一个Sql语句
  • 失效场景
  1. 事务进行了增删改操作且提交
  2. 手动清理了缓存

#{}和${}的区别:

  • #{}在底层使用PreparedStatement预编译为占位符,防止SQL注入
  • ${}在底层使用原始Statement直接进行字符串替换
对比项#{}${}
安全性高(防注入)低(需手动过滤)
底层实现预编译(PreparedStatement字符串替换(Statement
参数类型自动类型转换原样输出(字符串)
适用场景动态值(WHERE/INSERT/UPDATE)动态表名/列名/SQL 关键

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词