定义 为其他对象提供一种代理以控制这个对象的访问。 使用场景 职责清晰,真实角色就是实现业务逻辑,不必关系其他非本职的工作。 高扩展性,具体的角色随时变化,只要实现了接
定义
为其他对象提供一种代理以控制这个对象的访问。
使用场景
- 职责清晰,真实角色就是实现业务逻辑,不必关系其他非本职的工作。
- 高扩展性,具体的角色随时变化,只要实现了接口,代理类都可以在不用改变的情况下使用。
实现方式
一个简单的RPC通信,运用了动态代理。详细的代码请参考自己实现RPC通信
package com.fzsyw.client; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import com.fzsyw.domain.TransferMessage; public class RpcClient implements InvocationHandler { @SuppressWarnings("unchecked") public static <T> T proxy(Class<T> clazz) { return (T) Proxy.newProxyInstance(RpcClientTest.class.getClassLoader(), new Class[] {clazz}, new RpcClient(clazz)); } private Class<?> clazz; protected RpcClient(Class<?> clazz) { this.clazz = clazz; } /** * 代理逻辑 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { TransferMessage transferMessage = new TransferMessage(clazz.getName(), method.getName(), args); return new ClientSocket().transfer(transferMessage); } }
JDK动态代理的使用方式
- 使用
Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
获取代理对象 - 实现
InvocationHandler
接口,处理代理的逻辑 - 使用动态代理获取到
ITest
接口后,调用方式不变,只不过处理的逻辑交给了代理层,接口调用者感觉和普通接口调用一样。
package com.fzsyw.client; import com.fzsyw.domain.ITest; import com.fzsyw.domain.Person; public class RpcClientTest { public static void main(String[] args) { ITest test = RpcClient.proxy(ITest.class); System.out.println(test.say("hello", new Person("DD", 34))); } }
扩展与思考
- 代理模式可以让真实角色关注于业务逻辑,不受非业务逻辑的干扰。但是它映入了代理类,非核心业务在代理类中处理。在N层架构中,一些非核心业务逻辑可以引入代理层,在代理层中处理非核心业务逻辑。
- 动态代理的典型运用就是AOP,在Spring中实现方式主要有JDK代理和CGLIB代理。JDK代理需要接口,不能代理非
public
的方法,CGLIB代理需要继承实现类,不需要接口,不能代理final
和private
的方法。 - 强制代理,只有通过真实角色指定的代理才可以访问,也就是说真实角色管理代理角色。可以参考《设计模式之禅》代理模式章节。