代理模式
为其他对象提供一个代理以控制对某个对象的访问。代理类主要负责为真实对象提供 预处理消息、过滤消息、传递消息给委托类,代理类不现实具体服务,而是利用委托类来完成服务,并将执行结果封装处理。
其实就是代理类为被代理类预处理消息、过滤消息并在此之后将消息转发给被代理类,之后还能进行消息的 后置处理。代理类和被代理类通常会存在关联关系,代理类本身不实现服务,而是通过调用被代理类中的方法来提供服务。
代理模式可以大致分为两大部分,一是静态代理
,二是动态代理
。静态代理中,代理者的代码由程序员 自己 或 通过一些自动化工具生成固定的代码再对其进行编译,也就是说在我们的代码运行前代理类的class 编译文件就已经存在;而动态代理与静态代理相反,通过反射机制动态地生成代理者的对象,也就是说我们在 code 阶段压根就不需要知道代理谁,代理谁我们会在执行阶段决定。
静态代理
动态代理(JDK 中提供的工具)
1、动态代理的使用
Java 提供了一个便捷的动态代理接口 InvocationHandler
,还有 Proxy
工具类。
接口类定义:
public interface IPerson {
public abstract void walk();
}
实体对象定义:
public class Teacher implements IPerson {
@Override
public void walk() {
System.out.println(" walk : a teacher is walking ");
}
}
动态代理:
public class DynamicProxy implements InvocationHandler {
//被代理类(真实类)的引用
Object obj;
public DynamicProxy(Object obj){
this.obj = obj;
}
// 我们主要通过 invoke 方法来调用具体的被代理类方法,也就是真实的方法。
//动态代理使我们的代码逻辑更简洁
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("--动态代理:执行真实委托方法之前--");
Object result = method.invoke(obj, args);
System.out.println("--动态代理:执行真实委托方法之后--");
return result;
}
}
创建代理对象:
//创建真实对象
IPerson teacher = new Teacher();
//动态代理
DynamicProxy dp = new DynamicProxy(teacher);
/**
Proxy.newProxyInstance
根据指定的接口(一个或多个),返回一个代理对象。分配相关方法执行 到 指定的 InvocationHandler。
注意:第二个方法参数,必须是接口类型的Class对象
*/
//创建一个代理对象
IPerson proxyInstance = (IPerson) Proxy.newProxyInstance(teacher.getClass().getClassLoader(), new Class[]{IPerson.class}, dp);
//执行方法调用
proxyInstance.walk();
结果:
--动态代理:执行真实委托方法之前--
walk : a teacher is walking
--动态代理:执行真实委托方法之后--
2、动态代理的实现原理
原理:
通过 用户传入的ClassLoader、Class[] interfaces 和 和 InvocationHander 动态生成一个代理类的 Class 对象(这个类型是在运行期间确定的,所以原理上所有的类都可以在运行期间去编写😁)这个代理类 实现了 Class[] interfaces的所有抽象方法 和 Object 的 hashCode、equals 和 toString 方法。 interfaces 中的抽象方法的实现:直接调用 InvocationHandler 中的 invoke 方法。
看一个生成的代理类:
package com.sun.proxy;
import com.lakala.dyProxy.IPerson;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
public final class $Proxy0 extends Proxy implements IPerson {
private static Method m1;
private static Method m3;
private static Method m2;
private static Method m0;
public $Proxy0(InvocationHandler var1) throws {
super(var1);
}
public final boolean equals(Object var1) throws {
try {
return (Boolean)super.h.invoke(this, m1, new Object[]{var1});
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
}
public final void walk() throws {
try {
super.h.invoke(this, m3, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final String toString() throws {
try {
return (String)super.h.invoke(this, m2, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final int hashCode() throws {
try {
return (Integer)super.h.invoke(this, m0, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
static {
try {
m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
m3 = Class.forName("com.lakala.dyProxy.IPerson").getMethod("walk");
m2 = Class.forName("java.lang.Object").getMethod("toString");
m0 = Class.forName("java.lang.Object").getMethod("hashCode");
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}
过程
Proxy.newProxyInstance
这个方法会在运行期间动态生成 代理类型的类对象(Class)实例
,然后通过反射的方法,创建代理类的实例对象。
cglib (Code Generation Library)是一个开源项目。 是一个强大的,高性能,高质量的Code生成类库,它可以在运行期扩展Java类与实现Java接口。 JDK的动态代理用起来非常简单,但它有一个限制,就是使用动态代理的对象必须实现一个或多个接口。如果想代理没有实现接口的继承的类,该怎么办?现在我们可以使用CGLIB包。CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。
网友评论