专栏首页Java冰冻三尺Mybatis接口Mapper内的方法为啥不能重载?

Mybatis接口Mapper内的方法为啥不能重载?

> 公众号:[Java小咖秀](https://t.1yb.co/jwkk),网站:[javaxks.com](https://www.javaxks.com)

> 作者: 祖大俊 ,来源:https://my.oschina.net/zudajun/blog/666223

声明一个 interface 接口,没有任何实现类,却要求实例化接口对象,并能调用接口方法返回业务数据,老一辈 IT 革命家给出评论:这简直是无 -- 稽 --- 之谈。

一日小区漫步,我问朋友:Mybatis 中声明一个 interface 接口,没有编写任何实现类,Mybatis 就能返回接口实例,并调用接口方法返回数据库数据,你知道为什么不?朋友很是诧异:是啊,我也很纳闷,我们领导告诉我们按照这个模式编写就好了,我同事也感觉很奇怪,虽然我不知道具体是怎么实现的,但我觉得肯定是……(此处略去若干的漫天猜想),但是也不对啊,难道是……(再次略去若干似懂非懂)。

这激发了我写本篇文章的冲动。

动态代理的功能:通过拦截器方法回调,对目标 target 方法进行增强。

言外之意就是为了增强目标 target 方法。上面这句话没错,但也不要认为它就是真理,殊不知,动态代理还有投鞭断流的霸权,连目标 target 都不要的科幻模式。

注:本文默认认为,读者对动态代理的原理是理解的,如果不明白 target 的含义,难以看懂本篇文章,建议先理解动态代理。

1. 自定义 JDK 动态代理之投鞭断流实现自动映射器 Mapper

首先定义一个 pojo。

public class User {
    private Integer id;
    private String name;
    private int age;
​
    public User(Integer id, String name, int age) {
        this.id = id;
        this.name = name;
        this.age = age;
    }
    
}

再定义一个接口 UserMapper.java。

public interface UserMapper {
    public User getUserById(Integer id);    
}

接下来我们看看如何使用动态代理之投鞭断流,实现实例化接口并调用接口方法返回数据的。

自定义一个 InvocationHandler。

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
​
public class MapperProxy implements InvocationHandler {
​
    @SuppressWarnings("unchecked")
    public <T> T newInstance(Class<T> clz) {
        return (T) Proxy.newProxyInstance(clz.getClassLoader(), new Class[] { clz }, this);
    }
​
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if (Object.class.equals(method.getDeclaringClass())) {
            try {
                
                return method.invoke(this, args);
            } catch (Throwable t) {
            }
        }
        
        return new User((Integer) args[0], "zhangsan", 18);
    }
}

上面代码中的 target,在执行 Object.java 内的方法时,target 被指向了 this,target 已经变成了傀儡、象征、占位符。在投鞭断流式的拦截时,已经没有了 target。

写一个测试代码:

public static void main(String[] args) {
    MapperProxy proxy = new MapperProxy();
​
    UserMapper mapper = proxy.newInstance(UserMapper.class);
    User user = mapper.getUserById(1001);
​
    System.out.println("ID:" + user.getId());
    System.out.println("Name:" + user.getName());
    System.out.println("Age:" + user.getAge());
​
    System.out.println(mapper.toString());
}

output:

ID:1001
Name:zhangsan
Age:18
x.y.MapperProxy@6bc7c054

这便是 Mybatis 自动映射器 Mapper 的底层实现原理。

可能有读者不禁要问:你怎么把代码写的像初学者写的一样?没有结构,且缺乏美感。

必须声明,作为一名经验老道的高手,能把程序写的像初学者写的一样,那必定是高手中的高手。这样可以让初学者感觉到亲切,舒服,符合自己的 Style,让他们或她们,感觉到大牛写的代码也不过如此,自己甚至写的比这些大牛写的还要好,从此自信满满,热情高涨,认为与大牛之间的差距,仅剩下三分钟。

2. Mybatis 自动映射器 Mapper 的源码分析

首先编写一个测试类:

    public static void main(String[] args) {
        SqlSession sqlSession = MybatisSqlSessionFactory.openSession();
        try {
            StudentMapper studentMapper = sqlSession.getMapper(StudentMapper.class);
            List<Student> students = studentMapper.findAllStudents();
            for (Student student : students) {
                System.out.println(student);
            }
        } finally {
            sqlSession.close();
        }
    }

Mapper 长这个样子:

public interface StudentMapper {
    List<Student> findAllStudents();
    Student findStudentById(Integer id);
    void insertStudent(Student student);
}

org.apache.ibatis.binding.MapperProxy.java 部分源码。

public class MapperProxy<T> implements InvocationHandler, Serializable {
​
  private static final long serialVersionUID = -6424540398559729838L;
  private final SqlSession sqlSession;
  private final Class<T> mapperInterface;
  private final Map<Method, MapperMethod> methodCache;
​
  public MapperProxy(SqlSession sqlSession, Class<T> mapperInterface, Map<Method, MapperMethod> methodCache) {
    this.sqlSession = sqlSession;
    this.mapperInterface = mapperInterface;
    this.methodCache = methodCache;
  }
​
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    if (Object.class.equals(method.getDeclaringClass())) {
      try {
        return method.invoke(this, args);
      } catch (Throwable t) {
        throw ExceptionUtil.unwrapThrowable(t);
      }
    }
    
    final MapperMethod mapperMethod = cachedMapperMethod(method);
    return mapperMethod.execute(sqlSession, args);
  }

org.apache.ibatis.binding.MapperProxyFactory.java 部分源码。

public class MapperProxyFactory<T> {
​
  private final Class<T> mapperInterface;
​
  @SuppressWarnings("unchecked")
  protected T newInstance(MapperProxy<T> mapperProxy) {
    return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
  }

这便是 Mybatis 使用动态代理之投鞭断流

3. 接口 Mapper 内的方法能重载(overLoad)吗?(重要)

类似下面:

public User getUserById(Integer id);
public User getUserById(Integer id, String name);

Answer:不能。

原因:在投鞭断流时,Mybatis 使用 package+Mapper+method 全限名作为 key,去 xml 内寻找唯一 sql 来执行的。类似:key=x.y.UserMapper.getUserById,那么,重载方法时将导致矛盾。对于 Mapper 接口,Mybatis 禁止方法重载(overLoad)。

注:学习时,是先研究的源码,看懂了原理。写博文时,则先阐释原理,再阅读的源码。顺序刚好相反,希望读者不要因此疑惑,以为我强大到未卜先知。

原文链接:https://my.oschina.net/zudajun/blog/666223

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Mybatis接口Mapper内的方法为啥不能重载?

    来源:https://my.oschina.net/zudajun/blog/666223

    程序猿DD
  • 支付宝二面:Mybatis接口Mapper内的方法为啥不能重载吗?我直接懵逼了...

    言外之意就是为了增强目标target方法。上面这句话没错,但也不要认为它就是真理,殊不知,动态代理还有投鞭断流的霸权,连目标target都不要的科幻模式。

    开发者技术前线
  • 支付宝二面:Mybatis接口Mapper内的方法为啥不能重载吗?我直接懵逼了...

    言外之意就是为了增强目标target方法。上面这句话没错,但也不要认为它就是真理,殊不知,动态代理还有投鞭断流的霸权,连目标target都不要的科幻模式。

    后端码匠
  • MyBatis要不要学?京东内部的这份MyBatis文档真香!

    很多人说 mybatis重不重要?这里再说下,互联网公司基本都是用mybatis做为持久层框架的,所以mybatis是一定要学的。

    程序员追风
  • MyBatis要不要学?京东内部的这份MyBatis文档真香!

    MyBatis要不要学?京东内部的这份MyBatis文档真香! ...

    Java架构师必看
  • 深入理解Mybatis解析Mapper底层原理

    最近在使用高版本Spring Boot 2.x整合mybatis-plus 3.4.1时,控制台出现大量的warn提示XxxMapper重复定义信息:Bean ...

    用户4172423
  • 原创 | 从Spring Boot 2.x整合Mybatis-Plus深入理解Mybatis解析Mapper底层原理

    最近在使用高版本Spring Boot 2.x整合mybatis-plus 3.4.1时,控制台出现大量的warn提示XxxMapper重复定义信息:Bean ...

    猿芯
  • SpringBoot中关于Mybatis使用的三个问题

    上帝
  • SpringBoot中关于Mybatis使用的三个问题

    上帝
  • SpringBoot中关于Mybatis使用的三个问题

    转载请注明源地址:http://www.cnblogs.com/funnyzpc/p/8495453.html

    上帝
  • SpringBoot中关于Mybatis使用的三个问题

    转载请注明源地址:http://www.cnblogs.com/funnyzpc/p/8495453.html

    上帝
  • 为啥mybatis的mapper只有接口没有实现类,但它却能工作?

    说起mybatis,大伙应该都用过,有些人甚至底层源码都看过了。在mybatis中,mapper接口是没有实现类的,取而代之的是一个xml文件。也就是说我们调用...

    java思维导图
  • MyBatis-Plus 中 Mapper 重载踩坑指南

    近期在 Mapper 中写了个方法重载,然后死活查不到正确结果,最终灵机一动,想到是不是因为重载,然后我 Shift + F6 把重载方法名字改了一下!结果,显...

    程序员小航
  • Mybatis常见面试题总结及答案

    1、Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建stat...

    挨踢小子部落阁
  • 20+ 道常见的 MyBatis 面试题

    Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建statem...

    一个优秀的废人
  • MyBatis 常见面试题总结

    1、Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建stat...

    江南一点雨
  • 2020面试还搞不懂MyBatis?快看看这27道面试题!(含答案和思维导图)

    MyBatis是一个优秀的持久层ORM框架,它对jdbc的操作数据库的过程进行封装,使开发者只需要关注SQL 本身,而不需要花费精力去处理例如注册驱动、创建co...

    程序员追风
  • 答了Mybatis这个问题后,面试官叫我回去等通知……

    前段时间在我的技术群里,大家讨论起了为什么UserMapper.java是个接口,没有具体实现类,而我们可以直接调用其方法?

    田维常
  • MyBatis源码解析(二)——动态代理实现函数调用

    如果我们要使用MyBatis进行数据库操作的话,大致要做两件事情: 1. 定义DAO接口 在DAO接口中定义需要进行的数据库操作。 2. 创建映射文...

    大闲人柴毛毛

扫码关注云+社区

领取腾讯云代金券