前言:从“接口直接调用”的疑惑说起
使用过Mybatis的开发者都知道,在Mybatis中只需定义一个Mapper接口,无需编写实现类,就能直接调用其方法执行SQL操作。例如:
public interface UserMapper {
@Select("SELECT * FROM user WHERE id = #{id}")
User getUserById(int id);
}
// 直接调用接口方法(没有实现类!)
User user = userMapper.getUserById(1);
为什么接口不需要实现类就能工作?背后的核心机制正是Java动态代理。本文将深入剖析动态代理的实现原理,并解密Mybatis Mapper的“魔法”。
一、动态代理:Java的运行时魔法
1. 什么是动态代理?
动态代理是一种在程序运行时动态生成代理对象的技术,无需预先编写实现类。代理对象会拦截对目标方法的调用,并委托给InvocationHandler处理。
2. 两种实现方式
方式 | 原理 | 特点 |
---|---|---|
JDK动态代理 | 基于接口,利用Proxy 和InvocationHandler |
只能代理接口 |
CGLIB动态代理 | 通过继承目标类,生成子类代理 | 可代理类,需避免final方法 |
3. JDK动态代理示例
public interface HelloService {
void sayHello();
}
public class JdkProxyDemo {
public static void main(String[] args) {
HelloService proxy = (HelloService) Proxy.newProxyInstance(
HelloService.class.getClassLoader(),
new Class[]{
HelloService.class},
(proxy1, method, args1) -> {
System.out.println("Before method call");
return null; // 实际调用逻辑
});
proxy.sayHello(); // 输出:Before method call
}
}
关键点:
- 运行时生成名为
$Proxy0
的代理类 - 调用方法时转发到
InvocationHandler.invoke()