[Java]代理模式
【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)
https://www.cnblogs.com/cnb-yuchen/p/18002823
出自【进步*于辰的博客】
参考笔记一,P83。
代理模式分为静态代理和动态代理,比较简单,直接上示例,一目了然。为降低大家阅读成本,所有示例的打印结果都相同:
打开事务
转账金额:100
关闭事务
1
大家可自行cp测试。
静态代理
1:面向接口、目标类与代理类实现于同一接口。
// 目标类和代理类的公共接口
interface IService {
int transfer(int account);
}
// 目标类
class Target implements IService {
public int transfer(int account) {
System.out.println("转账金额:" + account);
return 1;
}
}
// 代理类
class Proxy implements IService {
private Target tar;
public Proxy(Target tar) {
this.tar = tar;
}
public int transfer(int account) {
System.out.println("打开事务");
int x = tar.transfer(account);
System.out.println("关闭事务");
return x;
}
}
class Test {
public static void main(String[] args) {
Proxy proxy = new Proxy(new Target());
int x = proxy.transfer(100);
System.out.println(x);
}
}
2:面向继承、代理类继承于目标类。
class Target {
public int transfer(int account) {
System.out.println("转账金额:" + account);
return 1;
}
}
class Proxy extends Target {
public int transfer(int account) {
System.out.println("打开事务");
int x = super.transfer(account);
System.out.println("关闭事务");
return x;
}
}
class Test {
public static void main(String[] args) {
Proxy proxy = new Proxy();
int x = proxy.transfer(100);
System.out.println(x);
}
}
动态代理
静态代理需要手动创建代理类,冗余。换个思路,反射可以根据Class信息创建实例,故可以通过反射为目标对象创建代理对象,则无需创建代理类,这就是“动态代理”。
JDK动态代理
面向接口,是JDK自带的。
interface IService {
int transfer(int account);
}
class Target implements IService {
public int transfer(int account) {
System.out.println("转账金额:" + account);
return 1;
}
}
class Test {
public static void main(String[] args) {
Target tar = new Target();
/**
* 通过 Proxy.newProxyInstance() 创建代理对象。
* 第一个参数是目标对象的类加载器,指定为哪个目标对象创建代理对象;
* 第二个参数是目标对象实现的接口,指定目标对象和代理对象的公共接口;
* 第三个参数是拦截器对象,指定用哪个拦截器来创建代理对象,需要实现 InvocationHandler 接口。
*/
// 由于 proxyInstance 是通过反射创建,存储于JVM,匿名(文末说明),故上转为 IService
IService proxyInstance = (IService) Proxy.newProxyInstance(tar.getClass().getClassLoader(),
tar.getClass().getInterfaces(),
new InvocationHandler() {
/**
* 代理(调用 transfer())时执行的方法
* @param proxy 代理对象,即 proxyInstance,暂不知如何使用
* @param method 目标对象的 Method 的 class 对象
* @param args 目标对象的 Method 的形参数组
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("打开事务");
// invoke() 是反射中 Method 对象执行时调用的方法,故动态代理是通过反射调用目标对象被代理的方法
Object result = method.invoke(tar, args);
System.out.println("关闭事务");
return result;
}
});
int x = proxyInstance.transfer(100);
System.out.println(x);
}
}
可以使用Lambda表达式省略new InvocationHandler()
。
Cglib动态代理
面向继承,基于Spring。
class Target {
public int transfer(int account) {
System.out.println("转账金额:" + account);
return 1;
}
}
// 动态代理类(拦截器)
class DynamicProxy implements MethodInterceptor {
private Object tar;
public DynamicProxy(Object tar) {
this.tar = tar;
}
public Object createProxy() {
Enhancer proxy = new Enhancer();// Enhancer 类是一种类生成器
proxy.setCallback(this);// 设置拦截器,即自身
proxy.setSuperclass(tar.getClass());// 设置父类,指定为哪个目标对象创建代理对象
return proxy.create();// 创建代理对象
}
/**
* 代理(调用 transfer())时执行的方法
* @param proxy 代理对象,即 proxyInstance,暂不知如何使用
* @param method 目标对象的 Method 的 class对象
* @param args 目标对象的 Method 的参数数组
* @param methodProxy 代理方法,即 Target.transfer(),暂不知如何使用
*/
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
System.out.println("打开事务");
Object result = method.invoke(tar, args);
System.out.println("关闭事务");
return result;
}
}
class Test {
public static void main(String[] args) {
Target target = new Target();
Target proxyInstance = (Target) new DynamicProxy(target).createProxy();
int x = proxyInstance.transfer(100);
System.out.println(x);
}
}
同样可以使用Lambda表达式优化,不过代理对象的创建(proxy.create()
)需要对Enhancer 类的属性进行一些设置,故进行了封装。
注意:两种动态代理都可以拦截所有方法,如:toString()
、hashcode()
。但不能拦截由final
修饰的方法,如:getClass()
。
扩展
JDK动态代理的核心是“面向接口”,实际上并不需要目标对象(公共接口的实现),也就是这样:
interface IService {
int transfer(int account);
}
class Test {
public static void main(String[] argx) {
Class z1 = IService.class;
IService proxyInstance = (IService) Proxy.newProxyInstance(z1.getClassLoader(),
new Class[] { z1 },
(proxy, method, args) -> {
System.out.println("打开事务");
System.out.println("关闭事务");
return 1;
});
int x = proxyInstance.transfer(100);
System.out.println(x);
}
}
没有目标对象,还是代理模式吗?又有什么意义?
Mybatis为Mapper接口创建代理对象使用的就是这种方式,不需要为Mapper接口创建实现类,我寻得一篇文章 → 一文搞懂Java动态代理:为什么Mybatis Mapper不需要实现类?(转发),那位博主讲述得狠详细,大家自行转站学习,我在此就不赘述了。
最后
本文中的例子是为了阐述静态代理和动态代理的实现思想、方便大家理解而简单举出的,不一定有实用性,仅是抛砖引玉。
本文完结。