静态代理和动态代理比较及案列

代理Proxy:

 

Proxy代理模式是一种结构型设计模式主要解决的问题是在直接访问对象时带来的问题

 

代理是一种常用的设计模式其目的就是为其他对象提供一个代理以控制对某个对象的访问代理类负责为委托类预处理消息过滤消息并转发消息以及进行消息被委托类执行后的后续处理


                      

 

为了保持行为的一致性代理类和委托类通常会实现相同的接口所以在访问者看来两者没有丝毫的区别通过代理类这中间一层能有效控制对委托类对象的直接访问也可以很好地隐藏和保护委托类对象同时也为实施不同控制策略预留了空间从而在设计上获得了更大的灵活性

 

更通俗的说代理解决的问题当两个类需要通信时引入第三方代理类将两个类的关系解耦让我们只了解代理类即可而且代理的出现还可以让我们完成与另一个类之间的关系的统一管理但是切记代理类和委托类要实现相同的接口因为代理真正调用的还是委托类的方法

 

使用场合举例

如果需要委托类处理某一业务那么我们就可以先在代理类中统一处理然后在调用具体实现类

 

按照代理的创建时期代理类可以分为两种 

静态由程序员创建代理类或特定工具自动生成源代码再对其编译在程序运行前代理类的.class文件就已经存在了

动态在程序运行时运用反射机制动态创建而成

 

下面分别用静态代理与动态代理演示一个示例

添加打印日志的功能即每个方法调用之前和调用之后写入日志

 

静态代理

 

具体用户管理实现类

[java] view plain copy

 print?

1. public class UserManagerImpl implements UserManager {  

2.   

3.     @Override  

4.     public void addUser(String userId, String userName) {  

5.         System.out.println("UserManagerImpl.addUser");  

6.     }  

7.   

8.     @Override  

9.     public void delUser(String userId) {  

10.         System.out.println("UserManagerImpl.delUser");  

11.     }  

12.   

13.     @Override  

14.     public String findUser(String userId) {  

15.         System.out.println("UserManagerImpl.findUser");  

16.         return "张三";  

17.     }  

18.   

19.     @Override  

20.     public void modifyUser(String userId, String userName) {  

21.         System.out.println("UserManagerImpl.modifyUser");  

22.   

23.     }  

24. }  

代理类--代理用户管理实现类

[java] view plain copy

 print?

1. public class UserManagerImplProxy implements UserManager {  

2.   

3.     // 目标对象  

4.     private UserManager userManager;  

5.     // 通过构造方法传入目标对象  

6.     public UserManagerImplProxy(UserManager userManager){  

7.         this.userManager=userManager;  

8.     }  

9.     @Override  

10.     public void addUser(String userId, String userName) {  

11.         try{  

12.                 //添加打印日志的功能  

13.                 //开始添加用户  

14.                 System.out.println("start-->addUser()");  

15.                 userManager.addUser(userId, userName);  

16.                 //添加用户成功  

17.                 System.out.println("success-->addUser()");  

18.             }catch(Exception e){  

19.                 //添加用户失败  

20.                 System.out.println("error-->addUser()");  

21.             }  

22.     }  

23.   

24.     @Override  

25.     public void delUser(String userId) {  

26.         userManager.delUser(userId);  

27.     }  

28.   

29.     @Override  

30.     public String findUser(String userId) {  

31.         userManager.findUser(userId);  

32.         return "张三";  

33.     }  

34.   

35.     @Override  

36.     public void modifyUser(String userId, String userName) {  

37.         userManager.modifyUser(userId,userName);  

38.     }  

39.   

40. }  


客户端调用

[java] view plain copy

 print?

1. public class Client {  

2.   

3.     public static void main(String[] args){  

4.         //UserManager userManager=new UserManagerImpl();  

5.         UserManager userManager=new UserManagerImplProxy(new UserManagerImpl());  

6.         userManager.addUser("1111""张三");  

7.     }  

8. }  

静态代理类优缺点

 

优点

 

代理使客户端不需要知道实现类是什么怎么做的而客户端只需知道代理即可解耦合),对于如上的客户端代码newUserManagerImpl()可以应用工厂将它隐藏如上只是举个例子而已

 

缺点

1)代理类和委托类实现了相同的接口代理类通过委托类实现了相同的方法这样就出现了大量的代码重复如果接口增加一个方法除了所有实现类需要实现这个方法外所有代理类也需要实现此方法增加了代码维护的复杂度

2)代理对象只服务于一种类型的对象如果要服务多类型的对象势必要为每一种对象都进行代理静态代理在程序规模稍大时就无法胜任了如上的代码是只为UserManager类的访问提供了代理但是如果还要为其他类如Department类提供代理的话就需要我们再次添加代理Department的代理类

 

举例说明代理可以对实现类进行统一的管理如在调用具体实现类之前需要打印日志等信息这样我们只需要添加一个代理类在代理类中添加打印日志的功能然后调用实现类这样就避免了修改具体实现类满足我们所说的开闭原则但是如果想让每个实现类都添加打印日志的功能的话就需要添加多个代理类以及代理类中各个方法都需要添加打印日志功能如上的代理方法中删除修改以及查询都需要添加上打印日志的功能

即静态代理类只能为特定的接口(Service)服务如想要为多个接口服务则需要建立很多个代理类

 

引入动态代理

 

根据如上的介绍你会发现每个代理类只能为一个接口服务这样程序开发中必然会产生许多的代理类

所以我们就会想办法可以通过一个代理类完成全部的代理功能那么我们就需要用动态代理

 

在上面的示例中一个代理只能代理一种类型而且是在编译器就已经确定被代理的对象而动态代理是在运行时通过反射机制实现动态代理并且能够代理各种类型的对象

 

Java中要想实现动态代理机制需要java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy 类的支持

 

java.lang.reflect.InvocationHandler接口的定义如下

[java] view plain copy

 print?

1. //Object proxy:被代理的对象  

2. //Method method:要调用的方法  

3. //Object[] args:方法调用时所需要参数  

4. public interface InvocationHandler {  

5.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable;  

6. }  

 

java.lang.reflect.Proxy类的定义如下

[java] view plain copy

 print?

1. //CLassLoader loader:类的加载器  

2. //Class<?> interfaces:得到全部的接口  

3. //InvocationHandler h:得到InvocationHandler接口的子类的实例  

4. public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException  


动态代理

 

具体实现类

[java] view plain copy

 print?

1. public class UserManagerImpl implements UserManager {  

2.   

3.     @Override  

4.     public void addUser(String userId, String userName) {  

5.         System.out.println("UserManagerImpl.addUser");  

6.     }  

7.   

8.     @Override  

9.     public void delUser(String userId) {  

10.         System.out.println("UserManagerImpl.delUser");  

11.     }  

12.   

13.     @Override  

14.     public String findUser(String userId) {  

15.         System.out.println("UserManagerImpl.findUser");  

16.         return "张三";  

17.     }  

18.   

19.     @Override  

20.     public void modifyUser(String userId, String userName) {  

21.         System.out.println("UserManagerImpl.modifyUser");  

22.   

23.     }  

24.   

25. }  


动态创建代理对象的类

[java] view plain copy

 print?

1. //动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类  

2.      

3. public class LogHandler implements InvocationHandler {  

4.   

5.     // 目标对象  

6.     private Object targetObject;  

7.     //绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。              

8.     public Object newProxyInstance(Object targetObject){  

9.         this.targetObject=targetObject;  

10.         //该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例    

11.         //第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器  

12.         //第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口  

13.         //第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandlerinvoke方法  

14.         //根据传入的目标返回一个代理对象  

15.         return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),  

16.                 targetObject.getClass().getInterfaces(),this);  

17.     }  

18.     @Override  

19.     //关联的这个实现类的方法被调用时将被执行  

20.     /*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/  

21.     public Object invoke(Object proxy, Method method, Object[] args)  

22.             throws Throwable {  

23.         System.out.println("start-->>");  

24.         for(int i=0;i<args.length;i++){  

25.             System.out.println(args[i]);  

26.         }  

27.         Object ret=null;  

28.         try{  

29.             /*原对象方法调用前处理日志信息*/  

30.             System.out.println("satrt-->>");  

31.               

32.             //调用目标方法  

33.             ret=method.invoke(targetObject, args);  

34.             /*原对象方法调用后处理日志信息*/  

35.             System.out.println("success-->>");  

36.         }catch(Exception e){  

37.             e.printStackTrace();  

38.             System.out.println("error-->>");  

39.             throw e;  

40.         }  

41.         return ret;  

42.     }  

43.   

44. }  

被代理对象targetObject通过参数传递进来我们通过targetObject.getClass().getClassLoader()获取ClassLoader对象然后通过targetObject.getClass().getInterfaces()获取它实现的所有接口然后将targetObject包装到实现了InvocationHandler接口的LogHandler对象中通过newProxyInstance函数我们就获得了一个动态代理对象

 

客户端代码

[java] view plain copy

 print?

1. public class Client {  

2.   

3.     public static void main(String[] args){  

4.         LogHandler logHandler=new LogHandler();  

5.         UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());  

6.         //UserManager userManager=new UserManagerImpl();  

7.         userManager.addUser("1111""张三");  

8.     }  

9. }  


可以看到我们可以通过LogHandler代理不同类型的对象如果我们把对外的接口都通过动态代理来实现那么所有的函数调用最终都会经过invoke函数的转发因此我们就可以在这里做一些自己想做的操作比如日志系统事务拦截器权限控制等这也就是AOP(面向切面编程)的基本原理

 

插曲

 

AOP(AspectOrientedProgramming):将日志记录性能统计安全控制事务处理异常处理等代码从业务逻辑代码中划分出来通过对这些行为的分离我们希望可以将它们独立到非指导业务逻辑的方法中进而改变这些行为的时候不影响业务逻辑的代码---解耦

 

针对如上的示例解释

 

我们来看上面的UserManagerImplProxy它的两个方法System.out.println("start-->addUser()")System.out.println("success-->addUser()"),这是做核心动作之前和之后的两个截取段正是这两个截取段却是我们AOP的基础OOPSystem.out.println("start-->addUser()")、核心动作System.out.println("success-->addUser()")这个三个动作在多个类里始终在一起但他们所要完成的逻辑却是不同的System.out.println("start-->addUser()")里做的可能是权限的判断在所有类中它都是做权限判断而在每个类里核心动作却各不相同System.out.println("success-->addUser()")可能做的是日志在所有类里它都做日志正是因为在所有的类里核心代码之前的操作和核心代码之后的操作都做的是同样的逻辑因此我们需要将它们提取出来单独分析设计和编码这就是我们的AOP思想一句话说AOP只是在对OOP的基础上进行进一步抽象使我们的类的职责更加单一

 

动态代理优点

 

动态代理与静态代理相比较最大的好处是接口中声明的所有方法都被转移到调用处理器一个集中的方法中处理InvocationHandler.invoke)。这样在接口方法数量比较多的时候我们可以进行灵活处理而不需要像静态代理那样每一个方法进行中转而且动态代理的应用使我们的类职责更加单一复用性更强

 

总结

 

其实所谓代理就是一个人或者一个机构代表另一个人或者另一个机构采取行动在一些情况下一个客户不想或者不能够直接引用一个对象而代理对象可以在客户端和目标对象之前起到中介的作用

 

代理对象就是把被代理对象包装一层在其内部做一些额外的工作比如用户需要上facebook,而普通网络无法直接访问网络代理帮助用户先翻墙然后再访问facebook。这就是代理的作用了

 

纵观静态代理与动态代理它们都能实现相同的功能而我们看从静态代理到动态代理的这个过程我们会发现其实动态代理只是对类做了进一步抽象和封装使其复用性和易用性得到进一步提升而这不仅仅符合了面向对象的设计理念其中还有AOP的身影这也提供给我们对类抽象的一种参考关于动态代理与AOP的关系个人觉得AOP是一种思想而动态代理是一种AOP思想的实现

猜你喜欢

转载自blog.csdn.net/qq_40016949/article/details/80207495