代理:一个角色代表别的一个角色来完成某些特定的功能。
比如:厂家,代理商,客户
客户买产品并不会直接与厂家打交道,客户只与代理商打交道,而代理商就可以对产品进行一些包装,搞一些促销及活动,提供一些售后的服务。
代理模式有三个角色: 1. 抽象主题角色 2. 代理主题角色 3. 实际被代理角色
其它类通过访问代理主题角色来访问实际被代理角色。
下面我们来个一个静态代理的实现。
我以一个坦克为例。
抽象主题角色:Moveable - package com.gjy.proxy;
- public interface Moveable {
- void move();
- }
复制代码 代理主题角色:TanktimeProxy - package com.gjy.proxy;
- public class TanktimeProxy implements Moveable{
- private Moveable t;
- public TanktimeProxy(Moveable t) {
- super();
- this.t = t;
- }
- @Override
- public void move() {
- long time1 = System.currentTimeMillis();
- System.out.println("time1="+time1);
- t.move();
- long time2 = System.currentTimeMillis();
- System.out.println("time2="+time2);
- System.out.println("运行时间为:"+(time2-time1));
- }
- }
复制代码 实际被代理对象:Tank - package com.gjy.proxy;
- public class Tank implements Moveable{
- @Override
- public void move() {
- System.out.println("TanK moving........");
- }
- }
复制代码 测试:- package com.gjy.proxy;
- public class TestTank {
- public static void main(String[] args) {
- Tank t = new Tank();
- Moveable move = new TanktimeProxy(t);
- move.move();
- }
- }
复制代码 从上例可以看到代理主题角色:TanktimeProxy实现了对Tank的move()方法运行时间的计算,而TanktimeProxy,Tank都实现了Moveable接口,通过调用TanktimeProxy的move()方法我们可以实现对Tank的move()方法的运行时间的计算,而不用在Tank的move()方法中作任何实现,这就是代理的作用。代理实现时TanktimeProxy,Tank必需实现Moveable接口。
下面我想在TanK的move()方法前后加上日志:
我必需再写一个类来实现这一功能: - package com.gjy.proxy;
- public class TanklogProxy implements Moveable{
- private Moveable t;
- public TanklogProxy(Moveable t) {
- super();
- this.t = t;
- }
- @Override
- public void move() {
- System.out.println("start move........");
- t.move();
- System.out.println("end move......");
- }
- }
复制代码 测试:- package com.gjy.proxy;
- public class TestTank {
- public static void main(String[] args) {
- Tank t = new Tank();
- Moveable move = new TanktimeProxy(t);
- Moveable movet = new TanklogProxy(move);
- movet.move();
- }
- }
复制代码 这样我通过代理在Tank的move()方法前后加入了日志和时间统计的功能,由于TanktimeProxy,TanklogProxy都实现了Moveable接口,所以TanklogProxy可以代理TanktimeProxy,反过来也可以,它们对Tank的代理顺序是可以交换的。
如果我想在Tank的move()方法调用的前后加入更多的功能,是不是要写更多的代理主题角色,这样子会使得代码过于臃肿,不易于维护,那有没有什么办法可以解决呢,答案是可以的,我们可以动态的来生成代理主题角色,来代理所有的被代理对象,这就是动态代理。
下面是一个简单的动态代理的实现:
类图如下:
|
|