【java基础】静态代理 VS 动态代理

来源:互联网 发布:乐视1s移动数据用不了 编辑:程序博客网 时间:2024/05/20 21:22

定义


       代理模式(Proxy Pattern)是对象的结构型模式,代理模式给某一个对象提供了一个代理对象,并由代理对象控制对原对象的引用。


       代理模式不会改变原来的接口和行为,只是转由代理干某件事,代理可以控制原来的目标,例如:代理商,代理商只会买东西,但并不会改变行为,不会制造东西。让我们通过下面的代码好好理解一下这句话。


分类


静态代理和动态代理


静态代理


静态代理类图

                


代码示例

     


接口

[java] view plaincopyprint?
  1. package com.liang.pattern;  
  2.   
  3. public interface UserManager {  
  4.   
  5.     public void addUser(String userId,String userName);  
  6.       
  7.     public void delUser(String userId);  
  8.       
  9.     public void modifyUser(String userId,String userName);  
  10.       
  11.     public String findUser(String userId);  
  12.       
  13. }  

目标对象
[java] view plaincopyprint?
  1. package com.liang.pattern;  
  2.   
  3. public class UserManagerImpl implements UserManager {  
  4.   
  5.     public void addUser(String userId, String userName) {  
  6.   
  7.         try{  
  8.             System.out.println("UserManagerImpl.addUser() userId-->>" + userId);  
  9.         }catch(Exception e){  
  10.             e.printStackTrace();  
  11.               
  12.             throw new RuntimeException();  
  13.         }  
  14.           
  15.           
  16.     }  
  17.   
  18.     public void delUser(String userId) {  
  19.   
  20.         System.out.println("UserManagerImpl.delUser() userId-->>" + userId);  
  21.     }  
  22.   
  23.     public String findUser(String userId) {  
  24.   
  25.         System.out.println("UserManagerImpl.findUser() userId-->>" + userId);  
  26.         return "于亮";  
  27.     }  
  28.   
  29.     public void modifyUser(String userId, String userName) {  
  30.   
  31.         System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);  
  32.     }  
  33.   
  34. }  

代理类,我们使用代理对象做一些日志记录,我们将简略的信息打印到控制台。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. package com.liang.pattern;  
  2.   
  3. public class UserManagerImplProxy implements UserManager {  
  4.   
  5.     private UserManager userManager;  
  6.     public UserManagerImplProxy(UserManager userManager){  
  7.         this.userManager = userManager;  
  8.     }  
  9.     public void addUser(String userId, String userName) {  
  10.                 //记录日志等操作或打印输入参数  
  11.         System.out.println("start-->>addUser() userId-->>" + userId);  
  12.         try{  
  13.   
  14.             userManager.addUser(userId, userName);  
  15.                 //执行成功,打印成功信息  
  16.             System.out.println("success-->>addUser()");  
  17.         }catch(Exception e){  
  18.             e.printStackTrace();  
  19.                 //失败时,打印失败信息  
  20.             System.out.println("error-->>addUser()");  
  21.             //throw new RuntimeException();  
  22.         }  
  23.     }  
  24.   
  25.     public void delUser(String userId) {  
  26.         //同上,略  
  27.         userManager.delUser(userId);  
  28.     }  
  29.   
  30.     public String findUser(String userId) {  
  31.         //同上,略  
  32.         userManager.findUser(userId);  
  33.         return null;  
  34.     }  
  35.   
  36.     public void modifyUser(String userId, String userName) {  
  37.         //同上,略  
  38.         userManager.modifyUser(userId, userName);  
  39.           
  40.     }  
  41.   
  42. }  

客户端调用
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. package com.liang.pattern;  
  2.   
  3. public class Client {  
  4.   
  5.     /** 
  6.      * @param args 
  7.      */  
  8.     public static void main(String[] args) {  
  9.         UserManager userManager = new UserManagerImplProxy(new UserManagerImpl());  
  10.         userManager.addUser("001","于亮");  
  11.     }  
  12. }   

输出结果,此方法执行成功
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. start-->>addUser() userId-->>001  
  2. UserManagerImpl.addUser() userId-->>001  
  3. success-->>addUser()  

       从类图我们可以看出,客户端本来可以直接和目标对象打交道,代理中间加了一个间接层,他们实现的功能是一样的,也没有改变参数。相信大家对上面的类图和代码很熟悉,跟我们平时看别人的博文一样,没有任何区别,下面我们看一下静态代理的优缺点。


优缺点


优点

1、直观感受,静态代理是实实在在的存在的,我们自己写的。

2、在编译期加入,提前就指定好了谁调用谁,效率高。


缺点

同样,它的优点也成了它致命的缺点。

1、静态代理很麻烦,需要大量的代理类

     当我们有多个目标对象需要代理时,我就需要建立多个代理类,改变原有的代码,改的多了就很有可能出问题,必须要重新测试。

2、重复的代码会出现在各个角落里,违背了一个原则:重复不是好味道

      我们应该杜绝一次次的重复。

3、在编译期加入,系统的灵活性差



       我们可以看到代理类的每个方法中,都有记录日志,执行成功或失败的代码,每个方法都重复了一遍,如果我们需要修改的话,并没有比不用静态代理时减少修改的地方,只是不用修改目标类。动态代理很好的为我们解决了这个问题,下面我们看一下动态代理。


动态代理


动态代理类图


               


代码示例



代理类(不明白,就看看注释吧)

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. package com.liang.pattern;  
  2.   
  3. import java.lang.reflect.InvocationHandler;  
  4. import java.lang.reflect.Method;  
  5. import java.lang.reflect.Proxy;  
  6. /** 
  7.  * 采用JDK动态代理必须实现InvocationHandler接口,采用Proxy类创建相应的代理类 
  8.  * @author liang 
  9.  * 
  10.  */  
  11. public class ProxyHandler implements InvocationHandler {  
  12.   
  13.     private Object targetObject;  
  14.     /** 
  15.      * 目标的初始化方法,根据目标生成代理类 
  16.      * @param targetObject 
  17.      * @return 
  18.      */  
  19.     public Object newProxyInstance(Object targetObject){  
  20.         this.targetObject = targetObject;  
  21.         //第一个参数,目标的装载器  
  22.         //第二个参数,目标接口,为每个接口生成代理  
  23.         //第三个参数,调用实现了InvocationHandler的对象,当你一调用代理,代理就会调用InvocationHandler的invoke方法  
  24.         return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);  
  25.     }  
  26.       
  27.     /** 
  28.      * 反射,这样你可以在不知道具体的类的情况下,根据配置的参数去调用一个类的方法。在灵活编程的时候非常有用。 
  29.      */  
  30.     public Object invoke(Object proxy, Method method, Object[] args)  
  31.             throws Throwable {  
  32.         //记录日志等操作或打印输入参数  
  33.         System.out.println("start-->>" + method.getName());  
  34.         for(int i=0;i<args.length;i++){  
  35.             //打印调用目标方法的参数  
  36.             System.out.println(args[i]);  
  37.         }  
  38.         Object ret = null;  
  39.         try{  
  40.             //调用目标方法  
  41.             ret = method.invoke(targetObject, args);  
  42.             //执行成功,打印成功信息  
  43.             System.out.println("success-->>" + method.getName());  
  44.         }catch(Exception e){  
  45.             e.printStackTrace();  
  46.             //失败时,打印失败信息  
  47.             System.out.println("error-->>" + method.getName());  
  48.             throw e;  
  49.         }  
  50.           
  51.         return ret;  
  52.     }  
  53.   
  54. }  

客户端调用
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. package com.liang.pattern;  
  2.   
  3. public class Client {  
  4.   
  5.     /** 
  6.      * @param args 
  7.      */  
  8.     public static void main(String[] args) {  
  9.   
  10.         ProxyHandler proxyHandler = new ProxyHandler();  
  11.         UserManager userManager = (UserManager)proxyHandler.newProxyInstance(new UserManagerImpl());  
  12.           
  13.         String name = userManager.findUser("0001");  
  14.         System.out.println("client.main-->>" + name);  
  15.     }  
  16.   
  17. }   


输出结果,运行成功

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. start-->>findUser  
  2. 0001  
  3. UserManagerImpl.findUser() userId-->>0001  
  4. success-->>findUser  
  5. client.main-->>于亮  


接口和目标类,同上,我就不再浪费大家的带宽了。


优缺点


优点

1、一个动态代理类更加简单了,可以解决创建多个静态代理的麻烦,避免不断的重复多余的代码

2、调用目标代码时,会在方法“运行时”动态的加入,决定你是什么类型,才调谁,灵活


缺点

1、系统灵活了,但是相比而言,效率降低了,比静态代理慢一点

2、动态代理比静态代理在代码的可读性上差了一点,不太容易理解

3、JDK动态代理只能对实现了接口的类进行代理


总结


       静态代理VS动态代理,打成了平手,各自有各的独特之处,均不可代替,在项目中到底使用哪种代理,没有最好,只有更合适。

0 0