Java Proxy

1、静态代理

 

 package ttitfly.proxy;   
   
 public interface HelloWorld {   
     public void print();   
 //  public void say();   
 } 

 

 package ttitfly.proxy;   
   
 public class HelloWorldImpl implements HelloWorld{   
   
     public void print(){   
         System.out.println("HelloWorld");   
     }   
 //  public void say(){   
 //      System.out.println("Say Hello!");   
 //  }   
 } 

 

 package ttitfly.proxy;   
   
 public class StaticProxy implements HelloWorld{   
   
     public HelloWorld helloWorld ;   
     public StaticProxy(HelloWorld helloWorld){   
         this.helloWorld = helloWorld;   
     }   
        
     public void print(){   
         System.out.println("Welcome");   
         //相当于回调   
         helloWorld.print();   
     }   
        
 //  public void say(){   
 //      //相当于回调   
 //      helloWorld.say();   
 //  }   
 }

 

 package ttitfly.proxy;   
   
 public class TestStaticProxy {   
   
     public static void main(String[] args){   
         HelloWorld helloWorld = new HelloWorldImpl();   
         StaticProxy staticProxy = new StaticProxy(helloWorld);   
         staticProxy.print();   
            
 //      staticProxy.say();   
     }   
 }

 可以看出静态代理类有一个很不爽的缺点:当如果接口加一个方法(把上面所有的代码的注释给去掉),所有的实现类和代理类里都需要做个实现。这就增加了代码的复杂度。动态代理就可以避免这个缺点。

 

二、动态代理

    动态代理与普通的代理相比较,最大的好处是接口中声明的所有方法都被转移到一个集中的方法中处理(invoke),这样,在接口方法数量比较多的时候,我们可以进行灵活处理,而不需要像静态代理那样每一个方法进行中转。

动态代理类只能代理接口,代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类

 

 package ttitfly.proxy;       
       
 import java.lang.reflect.InvocationHandler;       
 import java.lang.reflect.Method;       
 import java.lang.reflect.Proxy;       
 //动态代理类只能代理接口,代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类       
 public class DynamicProxy implements InvocationHandler{       
            
     private Object object;        
     //绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。   
     //Proxy.newProxyInstance的第三个参数是表明这些被拦截的方法执行时需要执行哪个InvocationHandler的invoke方法   
     public Object bindRelation(Object object){        
         this.object = object;       
         return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(),this);        
     }        
     //拦截关联的这个实现类的方法被调用时将被执行       
     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {        
         System.out.println("Welcome");       
         Object result = method.invoke(object, args);        
         return result;       
     }       
       
 }

 

 package ttitfly.proxy;       
       
 public class TestDynamicProxy {       
     public static void main(String[] args){       
         HelloWorld helloWorld = new HelloWorldImpl();       
         DynamicProxy dp = new DynamicProxy();       
         //在这里绑定的是HelloWorld,也就是HelloWorld是被代理接口。所以绑定关系时,需要传递一个HelloWorld的实现类的实例化对象。       
         HelloWorld helloWorld1 = (HelloWorld)dp.bindRelation(helloWorld);        
         helloWorld1.print();        
         helloWorld1.say();       
            
         //helloWorld2将不被拦截   
         HelloWorld helloWorld2 = new HelloWorldImpl();   
         helloWorld2.print();        
         helloWorld2.say();   
            
     }       
 } 

 

在测试类里调用实现类的print和say方法,因为代理类里代理了HelloWorld的所有方法。所以就不需要像静态代理类那样一一实现了。

 

附:代理模式 装饰模式 的区别

 

代理模式   可以在不改变接口的前提下     对过程进行控制


装饰模式    可以在不改变接口下     增强类的功能

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值