【Java后端】MyBatis-Plus 原理解析

MyBatis-Plus 原理解析

其实 MyBatis-Plus 的 Service 层设计 就是为了让开发者不用重复写很多样板代码。我们来一点点剖析 UserServiceImplIServiceUserServiceServiceImpl 之间的关系和调用链。


1. 类/接口关系图

复制代码
             IService<T>
                ▲
                │
        UserService (接口)   <-- 自定义的接口
                ▲
                │
 ServiceImpl<M, T> (抽象类,提供默认实现)
                ▲
                │
     UserServiceImpl (写的实现类)

2. 各个角色的作用

(1) IService<T>

  • MP 提供的 通用 Service 接口,定义了最常用的 CRUD 方法。
  • 例如:
java 复制代码
public interface IService<T> {
    boolean save(T entity);
    boolean removeById(Serializable id);
    boolean updateById(T entity);
    T getById(Serializable id);
    List<T> list();
    // ... 还有很多
}

(2) UserService

  • 你的业务接口 ,继承了 IService<User>
  • 这样 UserService 就自动拥有了所有 CRUD 方法,不用自己再声明。
  • 但你也可以在里面 加自定义业务方法
java 复制代码
public interface UserService extends IService<User> {
    User findByEmail(String email); // 自定义方法
}

(3) ServiceImpl<M, T>

  • MP 提供的 通用实现类 ,实现了 IService<T> 里的所有方法。
  • 内部通过调用 BaseMapper<T> 来完成数据库操作。

源码片段(简化):

java 复制代码
public class ServiceImpl<M extends BaseMapper<T>, T> implements IService<T> {
    @Autowired
    protected M baseMapper; // Mapper 注入进来

    @Override
    public boolean save(T entity) {
        return baseMapper.insert(entity) > 0;
    }

    @Override
    public T getById(Serializable id) {
        return baseMapper.selectById(id);
    }
}

(4) UserServiceImpl

  • 你的实现类 ,继承了 ServiceImpl<UserMapper, User> 并实现 UserService
  • 因为 ServiceImpl 已经实现了 IService 的方法,所以你啥都不用写,就自动拥有 savegetByIdlist 等方法。
  • 你可以在里面实现自定义的业务逻辑:
java 复制代码
@Service
public class UserServiceImpl 
    extends ServiceImpl<UserMapper, User> 
    implements UserService {

    @Override
    public User findByEmail(String email) {
        return lambdaQuery().eq(User::getEmail, email).one();
    }
}

3. 调用链条

比如调用 userService.save(user) 的时候,流程是这样的:

  1. Controller 调用
java 复制代码
userService.save(user);
  1. Spring 容器注入的其实是 UserServiceImpl 对象
    所以进入 UserServiceImpl.save()
  2. UserServiceImpl 继承自 ServiceImpl,没有重写 save(),于是调用的是 ServiceImpl.save()
  3. ServiceImpl.save() 内部会调用 baseMapper.insert(user) ,而 baseMapper 是注入的 UserMapper,继承了 BaseMapper<User>
  4. BaseMapper 底层最终由 MyBatis 执行 SQL,完成插入操作。

4. 关键总结

  • IService<T>定义通用方法(接口)。
  • UserService业务接口 ,扩展 IService<User>
  • ServiceImpl<M, T>默认实现 ,帮你实现了 IService 的所有方法。
  • UserServiceImpl实现类 ,继承 ServiceImpl,可以直接用 CRUD,也可以扩展业务方法。

所以:

👉 写的代码只需要关心业务逻辑,CRUD 全交给 MyBatis-Plus 的基类完成

相关推荐
沐浴露z22 分钟前
Kafka 生产者详解(上):消息发送流程与API,分区,吞吐量与数据可靠性
java·kafka·消息队列
程序员大雄学编程38 分钟前
「用Python来学微积分」16. 导数问题举例
开发语言·python·数学·微积分
float_com41 分钟前
【java面向对象进阶】------抽象类+接口 综合实例
java
哈哈哈哈~1 小时前
多线程之线程池
java·线程池
洛克大航海1 小时前
7-SpringCloud-服务网关 Gateway-高级特性 Route
java·spring cloud·gateway·route
Dreams_l1 小时前
redis中的数据类型
java·开发语言
梵得儿SHI1 小时前
Java IO 流详解:字符流(Reader/Writer)与字符编码那些事
java·开发语言·字符编码·工作原理·字符流·处理文本
盖世英雄酱581361 小时前
java深度调试【第三章内存分析和堆内存设置】
java·后端
太过平凡的小蚂蚁1 小时前
Kotlin 协程中常见的异步返回与控制方式(速览)
开发语言·前端·kotlin
bug攻城狮2 小时前
SaaS多租户架构实践:字段隔离方案(共享数据库+共享Schema)
mysql·架构·mybatis·springboot·1024程序员节