MyBatis 提供了两种级别的缓存:一级缓存(本地会话缓存)和二级缓存(全局缓存)。理解这两种缓存的机制对于优化应用程序性能非常重要。
### 一级缓存
一级缓存是默认开启的,它作用于 `SqlSession` 级别。当同一个 `SqlSession` 中执行相同的查询语句时,MyBatis 会首先检查本地缓存中是否有结果集,如果有则直接返回缓存的数据,不再向数据库发送查询请求。但是,一旦 `SqlSession` 关闭或提交事务后,这个缓存就失效了。
#### 注意:
- 一级缓存不会跨越多个 `SqlSession` 实例。
- 如果在同一个 `SqlSession` 中对数据进行了更新、插入或者删除操作,那么相应的查询缓存会被清空,以保证数据的一致性。
### 二级缓存
二级缓存可以跨多个 `SqlSession` 实例,甚至可以在整个应用范围内共享。为了启用二级缓存,你需要做以下几件事情:
1. **配置 MyBatis 的缓存设置**:在 MyBatis 配置文件中设置 `<setting name="cacheEnabled" value="true"/>` 来开启二级缓存功能,默认情况下它是开启的。
2. **为映射器启用缓存**:在 Mapper XML 文件中添加 `<cache/>` 或者更复杂的 `<cache>` 定义来指定缓存行为,比如你可以定义缓存的时间、大小限制等参数。
```xml
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
```
- `eviction`:指定了回收策略,如 FIFO(先进先出)、LRU(最近最少使用)、SOFT(软引用)或 WEAK(弱引用)。
- `flushInterval`:设定自动刷新的时间间隔(毫秒),超过此时间后缓存将被清空。
- `size`:设定缓存的最大条目数量。
- `readOnly`:是否只读。如果设置为 true,则不同 Session 之间的缓存是可共享且不可修改的;如果设置为 false,则允许每个 Session 对自己的副本进行修改。
3. **确保实体类实现了序列化接口**:因为二级缓存可能会涉及到对象的序列化与反序列化过程,所以要求存储到二级缓存中的所有对象都必须实现 Java 的 `Serializable` 接口。
4. **管理缓存刷新**:如果你的应用程序频繁地修改数据,你可能需要更加精细地控制缓存的刷新逻辑。可以通过配置 `<select>` 语句的 `flushCache` 属性以及 `<insert>`, `<update>` 和 `<delete>` 语句的 `flushCache` 属性来决定何时清除缓存。
### 缓存插件
除了上述内置的缓存机制外,MyBatis 还支持通过插件扩展自定义缓存。例如,可以集成第三方缓存库,如 Ehcache、Caffeine 或 Redis 等,来提供更强大和灵活的缓存解决方案。
### 最佳实践
- **评估需求**:并非所有的查询都需要缓存,应该根据实际业务场景选择性地启用缓存。
- **监控缓存效果**:定期检查缓存命中率和其他性能指标,以确保缓存设置合理并有效提升性能。
- **保持数据一致性**:特别注意避免因缓存而导致的数据不一致问题,尤其是在分布式系统环境中。
综上所述,MyBatis 的缓存机制可以帮助减少重复查询数据库的次数,从而提高应用的整体性能。正确配置和使用缓存对于构建高效稳定的持久层至关重要。