定义:为其他对象提供一个代理以控制对这个对象的访问。
类图:
说明:Subject表示抽象主题角色,它声明了真实主题和代理主题的共同接口,这样一来在任何使用真实主题的地方发都可以使用代理主题;Propxy表示主题角色,代理主题角色内部含有对真实主题的引用,从而可以在任何时候操作真实主题对象,代理主题角色提供一个与真实主题角色相同的接口,以便可以在任何时候都可以代替真实主题,代理角色通常在将客户端调用传递给真实的主题之前或之后,都要执行某个操作,而不是单纯的将调用传递给真实主题;RealSubject表示真是主题角色,它定义了代理角色所代表的真实对象。
优点:
①职责清晰,真实的角色就是实现实际的业务逻辑,不用担心其他非本职责的事物,通过后期的代理完成一件完成事务,附带的结果就是编程简洁清晰。
②代理对象可以在客户端和目标对象之间起到中接的作用,这样起到了中介的作用和保护了目标对象的引用。
③高扩展性
缺点:
①在客户端和目标对象增加了一个代理对象,会造成请求速度变慢
②增加系统的复杂性。
适用环境:
①远程代理:也就是为一个对象在不同的地址空间提供局部代表,这样可以隐藏一个对西安哪个存在于不同地址空间的事实。
②虚拟代理,根据需要创建开销很大的对象,通过它来存放实例化需要很长时间的对象。
③安全代理,用来控制真实对象访问时的权限。
③智能指引,当调用目标对象时,代理可以处理其他的一些操作。
实例场景:某应用软件中需要记录业务方法的调用日志,在不修改现有业务类的基础上为每一个类提供一个日志记录代理类,在代理类中输出日志。在代理类中调用真实业务类的业务方法,使用代理模式设计该日志记录模块的结构。
实例类图:
实例代码:
/**
* 抽象日志记录类:抽象主题
* @author fly
*
*/
interface AbstractLog {
public void method();
}
/**
* 业务类:真实主题
* @author fly
*
*/
class BusinessClass implements AbstractLog{
@Override
public void method() {
// TODO Auto-generated method stub
System.out.println("真实业务方法!");
}
}
/**
* 日志记录代理类:代理主题
* @author fly
*
*/
class LoggerProxy implements AbstractLog{
private BusinessClass business;
public LoggerProxy() {
// TODO Auto-generated constructor stub
business = new BusinessClass();
}
@Override
public void method() {
// TODO Auto-generated method stub
Calendar calendar = new GregorianCalendar();
int year = calendar.get(Calendar.YEAR);
int month = calendar.get(Calendar.MONTH);
int day = calendar.get(Calendar.DAY_OF_MONTH);
int hour = calendar.get(Calendar.HOUR)+12;
int minute = calendar.get(Calendar.MINUTE);
int second = calendar.get(Calendar.SECOND);
String dateTime = year+"-"+month+"-"+day+" "+hour+":"+minute+":"+second+"!";
System.out.println("方法method()被调用,调用时间为"+dateTime);
try {
business.method();
System.out.println("方法method()调用成功!");
} catch (Exception e) {
// TODO: handle exception
System.out.println("方法method()调用失败!");
}
}
}
//客户端测试
public class Client {
public static void main(String[] args) {
AbstractLog al;
al = new LoggerProxy();
al.method();
}
}
//运行结果:
以上是静态代理,动态代理demo,参考资料
1.JDK动态代理实现原理