深入浅出java静态代理和动态代理

首先介绍一下,什么是代理:

代理模式,是常用的设计模式。特征是,代理类与委托类有相同的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类。以及事后处理消息。

代理类和委托类,存在着关联关系。代理类的对象本身并不真正实现服务,知识通过调用委托类的对象的相关方法。

代理类可以分为两种:静态代理和动态代理。

静态代理:

代理类是由程序员创建,或由工具生成的代码 编译成的。在程序运行前,代理类的 *.class文件已经存在了。直接就可以运行 。

动态代理:

动态代理的代理类。没有直接由源代码生成。动态代理类的对象是在程序运行时由Java反射机制动态生成,不需要手工编写源代码。从而提高了软件的可扩展性。JAVA反射机制可以生成任意类型的动态代理类。

静态代理的实现:

接口:

  1. package test.static.pattern;
  2. public interface UserManager {
  3. public void addUser(String userId,String userName);
  4. public void delUser(String userId);
  5. public String findUser(String userId);
  6. public void modifyUser(String userId,String userNameString);
  7. }

实现类:

  1. package test.static.pattern;
  2. public class UserManagerImpl implements UserManager {
  3. @Override
  4. public void addUser(String userId, String userName) {
  5. System.out.println("UserManagerImpl.addUser() userId-->>" + userId);
  6. }
  7. @Override
  8. public void delUser(String userId) {
  9. System.out.println("UserManagerImpl.delUser() userId-->>" + userId);
  10. }
  11. @Override
  12. public String findUser(String userId) {
  13. System.out.println("UserManagerImpl.findUser() userId-->>" + userId);
  14. return null;
  15. }
  16. @Override
  17. public void modifyUser(String userId, String userNameString) {
  18. System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);
  19. }
  20. }

静态代理类:(只持有对象的引用)

  1. package test.static.pattern;
  2. public class UserManagerImplProxy implements UserManager {
  3. private UserManager userManager;
  4. public UserManagerImplProxy (UserManager userManager){
  5. this.userManager=userManager;
  6. }
  7. @Override
  8. public void addUser(String userId, String userName) {
  9. userManager.addUser(userId, userName);
  10. }
  11. @Override
  12. public void delUser(String userId) {
  13. userManager.delUser(userId);
  14. }
  15. @Override
  16. public String findUser(String userId) {
  17. return userManager.findUser(userId);
  18. }
  19. @Override
  20. public void modifyUser(String userId, String userNameString) {
  21. userManager.modifyUser(userId, userNameString);
  22. }
  23. }

客户端:

  1. package test.static.pattern;
  2. public class Client {
  3. /**
  4. * @param args
  5. */
  6. public static void main(String[] args) {
  7. //正常思路:客户端直接实例化出 子类的对象
  8. UserManager userManager=new UserManagerImpl();
  9. //静态代理:客户端实例化代理,通过代理取 子类的引用
  10. UserManager userManager=new UserManagerImplProxy(new UserManagerImpl());
  11. userManager.addUser("0001", "张三");
  12. }
  13. }

执行结果:

静态代理,因为代理类持有对象的引用,所以可以对其进行控制。

现象1:对于各个功能模块来说,都要建立对应的代理类。造成大量的代理类

现象2:但是如果对各个实现相同的控制,则需要重复写大量的代码。

为了避免重复代码出现多次,我们接着看什么是动态代理。

动态代理的实现:

接口:

  1. package test.dynamic.pattern;
  2. public interface UserManager {
  3. public void addUser(String userId,String userName);
  4. public void delUser(String userId);
  5. public String findUser(String userId);
  6. public void modifyUser(String userId,String userNameString);
  7. }

实现类:

  • package test.dynamic.pattern;
  • public class UserManagerImpl implements UserManager {
  • @Override
  • public void addUser(String userId, String userName) {
  • System.out.println("UserManagerImpl.addUser() userId-->>" + userId);
  • }
  • @Override
  • public void delUser(String userId) {
  • System.out.println("UserManagerImpl.delUser() userId-->>" + userId);
  • }
  • @Override
  • public String findUser(String userId) {
  • System.out.println("UserManagerImpl.findUser() userId-->>" + userId);
  • return null;
  • }
  • @Override
  • public void modifyUser(String userId, String userNameString) {
  • System.out.println("UserManagerImpl.modifyUser() userId-->>" + userId);
  • }
  • }

动态代理生成类:

  1. package test.dynamic.pattern;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. import java.lang.reflect.Proxy;
  5. import org.omg.CORBA.SystemException;
  6. import org.omg.CORBA.portable.InputStream;
  7. import org.omg.CORBA.portable.InvokeHandler;
  8. import org.omg.CORBA.portable.OutputStream;
  9. import org.omg.CORBA.portable.ResponseHandler;
  10. public class LogHandler implements InvocationHandler {
  11. /**
  12. * 调用引用对象类的方法抽象
  13. */
  14. @Override
  15. public Object invoke(Object proxy, Method method, Object[] args)
  16. throws Throwable {
  17. Object ret=null;
  18. try {
  19. //调用目标方法
  20. ret=method.invoke(targetObject, args);
  21. } catch (Exception e) {
  22. throw e;
  23. }
  24. return ret;
  25. }
  26. //目标实现的引用
  27. private Object targetObject;
  28. /**
  29. * 代理类生成方法
  30. * @param targetObject
  31. * @return
  32. */
  33. public Object newProxyInstance(Object targetObject){
  34. this.targetObject=targetObject;
  35. return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
  36. }
  37. }

客户端:

  • package test.dynamic.pattern;
  • public class Client {
  • /**
  • * @param args
  • */
  • public static void main(String[] args) {
  • //动态代理
  • LogHandler logHandler=new LogHandler();
  • UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());
  • //userManager.addUser("0001", "张三");
  • userManager.delUser("111");
  • }
  • }

执行结果:

动态代理类,在程序中没有体现。只有在程序运行的时候采用创建相应的代理类。这样就可以少写大量的代理类。

原文发布于微信公众号 - Java团长(javatuanzhang)

原文发表时间:2017-04-20

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏鸿的学习笔记

sql解析的一些计划

关于sql解析的一些概述: 因为最近在研究如何将oracle的sql语句迁移到hive上去,前期是准备写一些udf函数去弥补hive缺失oracle函数...

872
来自专栏Hadoop数据仓库

HAWQ技术解析(十) —— 过程语言

        HAWQ支持用户自定义函数(user-defined functions,UDF),还支持给HAWQ内部的函数起别名。编写UDF的语言可以是SQ...

4745
来自专栏扎心了老铁

hiveQL求差集

hive sql求差集的方法 1、什么是差集 set1 - set2,即去掉set1中存在于set2中的数据。 2、hive中计算差集的方法,基本是使用左外链接...

4744
来自专栏AhDung

【SQL】小心字符串拼接导致长度爆表

别想当然以为它会返回8002,而是8000,select @max也只会得到8000个a,后面两个b没了。我们知道,varchar(max)类型不受字符数限制,...

1753
来自专栏菩提树下的杨过

mybatis的物理分页:mybatis-paginator

文件分割与合并是一个常见需求,比如:上传大文件时,可以先分割成小块,传到服务器后,再进行合并。很多高大上的分布式文件系统(比如:google的GFS、taoba...

2279
来自专栏Android机器圈

Java-单例模式详解(图文并茂,简单易懂)

PS:首先我们要先知道什么是单例,为什么要用单例,用的好处是什么等问题来看。 1:java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍两...

34911
来自专栏钟绍威的专栏

递归思想解决输出目录下的全部文件

刚刚了解了下递归思想 递归就是在方法内调用本方法 下面说一个实际的应用 输出目录下的全部文件,当目录中还有目录时,则进入目录输出里面的文件 import j...

1977
来自专栏Danny的专栏

未经处理的异常在 System.Data.dll 中发生。其他信息:在应使用条件的上下文(在 '***' 附近)中指定了非布尔类型的表达式。

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

2182
来自专栏LhWorld哥陪你聊算法

Hadoop源码篇---解读Mapprer源码outPut输出

上次讲完MapReduce的输入后,这次开始讲MapReduce的输出。注意MapReduce的原语很重要:

1313
来自专栏Danny的专栏

【MyBatis框架点滴】——MyBatis输出映射

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

1552

扫码关注云+社区

领取腾讯云代金券