【杂类】理解 @Repository 和 Mapper 的关系

目录

[​1. @Mapper(MyBatis 注解)​​](#1. @Mapper(MyBatis 注解))

[​2. @Repository(Spring 注解)​​](#2. @Repository(Spring 注解))

[​3. 二者在 MyBatis 项目中的协作关系​](#3. 二者在 MyBatis 项目中的协作关系)

[​场景 1:使用 @MapperScan(推荐)​​](#场景 1:使用 @MapperScan(推荐))

[场景 2:单独使用 @Mapper(不推荐)​​](#场景 2:单独使用 @Mapper(不推荐))

[​场景 3:同时使用 @Mapper+ @Repository(冗余)​​](#场景 3:同时使用 @Mapper+ @Repository(冗余))

[​4. 关系总结​](#4. 关系总结)

[​实践(Spring Boot + MyBatis)​​](#实践(Spring Boot + MyBatis))


1. @Mapper(MyBatis 注解)​

  • 作用:​

    标记在 ​Mapper 接口 上,告诉 ​MyBatis​:"这个接口需要被扫描并生成动态代理实现类"。

  • 责任方:​

    MyBatis 框架(通过 mybatis-spring整合包)。

  • 关键点:​

    • 必须配合 ​**@MapperScan** ​ 或 ​XML 配置使用,否则 MyBatis 不会扫描该接口。

    • 不涉及 Spring 容器管理​(单纯标记接口,不注册 Bean)。

    • 主要解决 ​​"MyBatis 如何识别哪些接口需要生成代理类"​​ 的问题。


2. @Repository(Spring 注解)​

  • 作用:​

    标记在 ​ 上(通常是 DAO 实现类),告诉 ​Spring​:"这是一个数据访问层组件,请注册为 Bean 并启用异常转换"。

  • 责任方:​

    Spring Framework。

  • 关键点:​

    • 属于 Spring 组件扫描的一部分(如 @ComponentScan)。

    • 自动启用 ​异常转换 ​(将 MyBatis/JDBC 异常转为 Spring DataAccessException)。

    • 主要解决 ​​"Spring 如何管理数据访问层 Bean"​​ 的问题。


3. 二者在 MyBatis 项目中的协作关系

场景 1:使用 @MapperScan(推荐)​
复制代码
@SpringBootApplication
@MapperScan("com.example.mapper") // MyBatis 扫描包
public class App { ... }
  • Mapper 接口:​

    复制代码
    // 无需加 @Repository!MyBatis 代理类已由 @MapperScan 注册为 Spring Bean
    public interface UserMapper {
        User selectById(Long id);
    }
  • 为什么不需要 @Repository?​

    @MapperScan会:

    1. 扫描指定包下的接口;

    2. 让 MyBatis 为这些接口生成动态代理类

    3. 自动将这些代理类注册为 Spring Bean ​(相当于完成了 @Repository的注册功能)。

    4. 自动启用异常转换 ​(整合包已处理,无需 @Repository)。

场景 2:单独使用 @Mapper(不推荐)​
复制代码
@Mapper // 仅标记接口,但未配置扫描
public interface UserMapper { ... }
  • 问题:​

    若未配置 @MapperScan,MyBatis 不会生成代理类,Spring 也无法注册 Bean → ​运行时报 NoSuchBeanDefinitionException

  • 解决方案:​

    必须添加 @MapperScan或在配置文件中指定 Mapper 包路径。

场景 3:同时使用 @Mapper+ @Repository(冗余)​
复制代码
@Mapper
@Repository // 多余的!
public interface UserMapper { ... }
  • 效果:​

    • @Mapper让 MyBatis 识别接口;

    • @Repository会被 Spring 扫描到,但注册的是接口本身​(不是代理类),导致注入失败。

  • 结论:​

    不要同时在 Mapper 接口上加 @Repository!​

    它既无法替代 @Mapper的代理生成功能,又会干扰 Spring 对 MyBatis 代理 Bean 的识别。


4. 关系总结

注解 框架 作用对象 主要功能 是否必需
​**@Mapper**​ MyBatis 接口 标记需生成代理的 Mapper 接口 可选(可用 @MapperScan替代)
​**@Repository**​ Spring 注册 DAO Bean + 异常转换 在 MyBatis 中不需要
​**@MapperScan**​ MyBatis-Spring 扫描接口 → 生成代理 → 注册 Bean 必需 ​(推荐替代 @Mapper

实践(Spring Boot + MyBatis)​

  1. 配置类添加 @MapperScan:​

    复制代码
    @SpringBootApplication
    @MapperScan("com.example.mapper") // 指定 Mapper 接口包
    public class App { ... }
  2. Mapper 接口无需任何注解:​

    复制代码
    // 不加 @Mapper 或 @Repository!
    public interface UserMapper {
        @Select("SELECT * FROM user WHERE id = #{id}")
        User selectById(Long id);
    }
  3. Service 层直接注入:​

    复制代码
    @Service
    public class UserService {
        @Autowired
        private UserMapper userMapper; // 注入的是 MyBatis 代理对象
    }

关键结论​:

  • ​**@Mapper是 MyBatis 的"接口标识"​** ​(可被 @MapperScan替代);

  • ​**@Repository在 MyBatis 场景下完全不需要** ​(由 @MapperScan代理注册 + 异常转换);

  • 二者不要混用,避免冲突。

相关推荐
高山上有一只小老虎9 分钟前
等差数列前n项的和
java·算法
rockmelodies13 分钟前
东方通安装
java
避避风港39 分钟前
Java 抽象类
java·开发语言·python
初学小白...42 分钟前
JVM入门知识点
java·服务器·jvm
C++chaofan1 小时前
基于session实现短信登录
java·spring boot·redis·mybatis·拦截器·session
摇滚侠1 小时前
idea 刷新maven,提示java.lang.RuntimeException: java.lang.OutOfMemoryError
java·maven·intellij-idea
果壳~1 小时前
【Java】使用国密2,3,4.仿照https 统一请求响应加解密
java·https
N 年 后1 小时前
单独Docker部署和Docker Compose部署
java·docker·容器
lkbhua莱克瓦241 小时前
Java练习——数组练习
java·开发语言·笔记·github·学习方法
趙卋傑1 小时前
常见排序算法
java·算法·排序算法