代理模式是一种常见的设计模式,它使用代理对象来完成用户请求,屏蔽用户对真实对象的访问。特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。
主要用途:
1. 安全原因,需要屏蔽客户端直接访问真实对象;
2. 远程调用,需要代理类处理远程方法调用的技术细节;
3. 为了提升系统性能,对真实对象进行封装,达到延迟加载的目的,从而提升系统性能和反映速度。
一.uml图
二.静态代理
由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。
public interfaceDatabaseQuery {
public voidquery();
}
public classIbatisDatabaseQuery implements DatabaseQuery{
@Override
public voidquery() {
System.out.printf("doquery... \n");
}
}
public classDatabaseQueryProxy implements DatabaseQuery {
private DatabaseQuery dbQuery;
public DatabaseQueryProxy(DatabaseQuerydatabaseQuery) {
this.dbQuery =databaseQuery;
}
@Override
public voidquery() {
System.out.printf("beforequery ... \n");
dbQuery.query();
System.out.printf("afterquery ... \n");
}
public DatabaseQuery getDbQuery() {
return dbQuery;
}
public voidsetDbQuery(DatabaseQuery dbQuery) {
this.dbQuery =dbQuery;
}
}
public class Client {
public static void main(String[] args) {
DatabaseQuerydbQuery = new IbatisDatabaseQuery();
DatabaseQueryProxyproxy = new DatabaseQueryProxy(dbQuery);
proxy.query();
}
}
三.动态代理
在程序运行时,运用反射机制动态创建而成。即,代理类的字节码将在运行时生成并载入当前的ClassLoader。生成动态代理类的方法很多,包括jdk自带的动态代理,cglib,javassist或asm。jdk的动态代理使用最为简单,不需引入额外的第三方库,但是功能较弱。cglib和javassist都是高级的字节码生成工具,总体性能比jdk自带的动态代理性能高。asm是低级字节码生成工具,使用asm已经近乎在使用java bytecode编程,过于繁琐,且性能没有数量级的提升,可维护性较差。
1. jdk方式
public class JdkProxyHandlerimplements InvocationHandler {
private DatabaseQuery dbQuery;
public JdkProxyHandler(DatabaseQuery dbQuery) {
this.dbQuery =dbQuery;
}
public DatabaseQuery bind() {
return (DatabaseQuery) Proxy.newProxyInstance(dbQuery.getClass().getClassLoader(), dbQuery.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method,Object[] args)
throws Throwable {
Objectresult = null;
System.out.printf("beforequery ... \n");
dbQuery.query();
System.out.printf("afterquery ... \n");
return result;
}
public DatabaseQuery getDbQuery() {
return dbQuery;
}
public voidsetDbQuery(DatabaseQuery dbQuery) {
this.dbQuery =dbQuery;
}
}
public class Client {
public static void main(String[] args) {
JdkProxyHandlerproxyHandler = new JdkProxyHandler(dbQuery);
proxyHandler.bind().query();
}
}