Java代理模式的深入了解

作者:空山新雨后~ 时间:2021-06-06 10:24:38 

一、静态代理模式

1.1、 代理模式的定义:

由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。

比如在有些情况下,一个客户不能或者不想直接访问另一个对象,这时需要找一个中介帮忙完成某项任务,这个中介就是代理对象。例如,购买火车票不一定要去火车站买,可以通过 12306 网站或者去火车票代售点买。又如找女朋友、找保姆、找工作等都可以通过找中介完成。

        Java代理模式的深入了解

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

代码实例:实现增删改查操作,通过代理

接口:

package com.proxyPattern.staticProxy2;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className UserService
* @date 2021/12/27 17:54
* @Description 服务接口
*/
public interface UserService {
   void add();
   void delete();
   void update();
   void query();
}

真实类(这里是服务类)

package com.proxyPattern.staticProxy2;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className UserServiceImp
* @date 2021/12/27 17:55
* @Description 服务实现类
*/
public class UserServiceImp implements UserService{
   @Override
   public void add() {
       System.out.println("添加了一条数据");
   }
   @Override
   public void delete() {
       System.out.println("删除了一条数据");
   }
   @Override
   public void update() {
       System.out.println("修改了一条数据");
   }
   @Override
   public void query() {
       System.out.println("查询了一条数据");
   }
}

代理类

package com.proxyPattern.staticProxy2;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className UserServiceProxy
* @date 2021/12/27 17:56
* @Description 服务代理类
*/
public class UserServiceProxy implements UserService {
   private UserServiceImp userServiceImp;
   public UserServiceProxy() {
   }
   public void setUserServiceImp(UserServiceImp userServiceImp) {
       this.userServiceImp = userServiceImp;
   }
   @Override
   public void add() {
       getLog("add");
       userServiceImp.add();
   }
   @Override
   public void delete() {
       getLog("delete");
       userServiceImp.delete();
   }
   @Override
   public void update() {
       getLog("update");
       userServiceImp.update();
   }
   @Override
   public void query() {
       getLog("add");
       userServiceImp.query();
   }
   public void getLog(String message) {
       System.out.println("日志:" + message + "语句执行了");
   }
}

客户端测试类

package com.proxyPattern.staticProxy2;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className Customer
* @date 2021/12/27 18:00
* @Description 客户终端测试类
*/
public class Customer {
   public static void main(String[] args) {
       UserServiceImp userServiceImp = new UserServiceImp();
       UserServiceProxy p = new UserServiceProxy();
       p.setUserServiceImp(userServiceImp);
       p.add();
       p.update();
       p.delete();
       p.query();
   }
}
/**
* 执行结果:
* 日志:add语句执行了
* 添加了一条数据
* 日志:update语句执行了
* 修改了一条数据
* 日志:delete语句执行了
* 删除了一条数据
* 日志:add语句执行了
* 查询了一条数据
*/

        上述代码看到我们并没有使用userServiceImp去执行方法,而是使用了一个代理类去执行,这就是代理模式,类似于你租房并没有找房东租房,而是找的一个中间代理人中介来完成租房这个动作。

1.2、代理模式的优缺点

那么代理模式有哪些优点呢?

1、可以使得我们的真实角色更加纯粹 ,不再去关注一些公共的事情 
2、公共的业务由代理来完成 . 实现了业务的分工 
3、公共业务发生扩展时变得更加集中和方便 

缺点

1、代理模式会造成系统设计中类的数量增加

2、在客户端和目标对象之间增加一个代理对象,会造成请求处理速度变慢;

3、增加了系统的复杂度;

 如何解决这些问题呢?就靠下面的 * 模式来解决

二、 * 模式

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

没错,动态的代理模式使用的是反射,而且要自己写一个 * 类去动态的获取一个代理类

代码实例:案例同上,只不过采用的是 * 模式

服务实现类(真实类)

package com.proxyPattern.staticProxy2;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className Customer
* @date 2021/12/27 18:00
* @Description 客户终端测试类
*/
public class Customer {
   public static void main(String[] args) {
       UserServiceImp userServiceImp = new UserServiceImp();
       UserServiceProxy p = new UserServiceProxy();
       p.setUserServiceImp(userServiceImp);
       p.add();
       p.update();
       p.delete();
       p.query();
   }
}
/**
* 执行结果:
* 日志:add语句执行了
* 添加了一条数据
* 日志:update语句执行了
* 修改了一条数据
* 日志:delete语句执行了
* 删除了一条数据
* 日志:add语句执行了
* 查询了一条数据
*/

接口:

package com.proxyPattern. autoProxy;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className UserService
* @date 2021/12/27 17:54
* @Description 服务接口
*/
public interface UserService {
   void add();
   void delete();
   void update();
   void query();
}

* 类,这个几乎可以做一个工具类使用,因为格式固定        

package com.proxyPattern.autoProxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.autoProxy
* @className ProxyInvocationHandler
* @date 2021/12/27 19:33
* @Description * 类
*/
public class ProxyInvocationHandler implements InvocationHandler {
   //被代理的接口
   private Object target;
   public void setTarget(Object target) {
       this.target = target;
   }
   /**
    * @Date  2021/12/27 19:36
    * @Param
    * @Return Object
    * @MetodName getProxy
    * @Author wang
    * @Description 生成得到代理类
    */
   public Object getProxy() {
       return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
   }
   /**
    * @Date  2021/12/27 19:34
    * @Param
    * @param proxy
    * @param method
    * @param args
    * @Return Object
    * @MetodName invoke
    * @Author wang
    * @Description 处理代理实例,并返回结果
    */
   @Override
   public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
       log(method.getName());
       Object result = method.invoke(target, args);
       return result;
   }
   public void log(String message) {
       System.out.println("日志:" + message + "语句执行了" );
   }
}

客户端测试类:

package com.proxyPattern.autoProxy;
/**
* @author wang
* @version 1.0
* @packageName com.proxyPattern.staticProxy2
* @className Customer
* @date 2021/12/27 18:00
* @Description 客户终端测试类
*/
public class Customer {
   public static void main(String[] args) {
      //真实角色
       UserService userService = new UserServiceImp();
       //代理角色
       ProxyInvocationHandler pih = new ProxyInvocationHandler();
       //动态设置代理的对象
       pih.setTarget(userService);
       //动态生成代理类
       UserService proxy = (UserService) pih.getProxy();
       proxy.query();
       proxy.update();
   }
}
/**
* 日志:query语句执行了
* 查询了一条数据
* 日志:update语句执行了
* 修改了一条数据
*/

可以看到我们这里可以更方便的去获取代理类了,只需要将动态设置代理类那里的对象改一下,就可以去代理别的类。

来源:https://blog.csdn.net/ww166955/article/details/122179161

标签:Java,代理,模式
0
投稿

猜你喜欢

  • Android程序美化之自定义ListView背景的方法

    2023-12-13 17:52:57
  • UnityShader使用速度映射图实现运动模糊

    2021-11-23 03:53:21
  • java创建子类对象设置并调用父类的变量操作

    2023-10-13 18:16:13
  • MyBatis-plus实现逆向生成器

    2021-08-26 16:37:14
  • Android Xutils3网络请求的封装详解及实例代码

    2021-07-20 20:27:49
  • Java中去除字符串中所有空格的几种方法

    2023-11-24 04:59:24
  • springMVC引入Validation的具体步骤详解

    2021-12-19 18:30:19
  • 详解C# Socket简单例子(服务器与客户端通信)

    2022-01-02 03:12:49
  • FFmpeg Principle分析Out put File 数据结构

    2023-02-17 08:26:49
  • c# 引用类型和值类型

    2023-10-11 08:20:48
  • java实现的冒泡排序算法示例

    2022-12-29 20:35:03
  • 解析Flink内核原理与实现核心抽象

    2023-11-25 10:30:04
  • JavaSE的逻辑控制你了解吗

    2023-01-07 01:32:22
  • android webView截图的4种方法

    2023-02-04 19:54:12
  • Spring Boot如何通过自定义注解实现日志打印详解

    2022-07-04 14:10:18
  • 深入浅析Spring 的aop实现原理

    2023-01-10 00:00:10
  • Java实现提取QSV文件视频内容

    2023-08-24 13:33:45
  • Android Studio 3.3.2 正式版的安装教程图解

    2022-01-13 22:22:34
  • Spring Security 实现用户名密码登录流程源码详解

    2023-05-31 11:13:59
  • SpringBoot如何读取xml配置bean(@ImportResource)

    2021-08-03 22:33:44
  • asp之家 软件编程 m.aspxhome.com