当前位置: 首页 > news >正文

研磨设计模式day11代理模式

目录

场景

代码实现

​编辑

解析

定义

代理模式调用示意图

代理模式的特点 

本质 

​编辑何时选用 


场景

我有一个订单类,包含订单数、用户名和商品名,有一个订单接口包含了对订单类的getter和setter

现在有一个需求,a创建的订单只能a修改,其他人无权修改

代码实现

定义OrderApi接口

package day10代理模式;public interface OrderApi {public String getProductName();void setProductName(String productName,String user);public String getOrderUser();public void setOrderUser(String orderUser,String user);public int getOrderNum();void setOrderNum(int orderNum,String user);
}

Order类,实现订单接口

package day10代理模式;public class Order implements OrderApi{private String productName;private int orderNum;private String orderUser;public Order(String productName, int orderNum, String orderUser) {this.productName = productName;this.orderNum = orderNum;this.orderUser = orderUser;}@Overridepublic String toString() {return "Order{" +"productName='" + productName + '\'' +", orderNum=" + orderNum +", orderUser='" + orderUser + '\'' +'}';}public void setOrderUser(String orderUser) {this.orderUser = orderUser;}public void setProductName(String productName) {this.productName = productName;}@Overridepublic int getOrderNum() {return orderNum;}@Overridepublic void setOrderNum(int orderNum,String user) {this.orderNum = orderNum;}@Overridepublic String getProductName() {return productName;}@Overridepublic void setProductName(String productName, String user) {this.productName = productName;}@Overridepublic String getOrderUser() {return this.orderUser;}@Overridepublic void setOrderUser(String orderUser, String user) {this.orderUser = orderUser;}
}

现在创建一个代理类

package day10代理模式.java中的代理类;import day10代理模式.Order;
import day10代理模式.OrderApi;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;/*** 使用java中的动态代理*/
public class DynamicProxy implements InvocationHandler {/*** 被代理的对象*/private OrderApi order = null;/*** 获取 绑定好代理和具体对象后的目标对象的 接口** @param order 具体的订单对象,相当于具体目标对象* @return 绑定好代理和具体对象后的目标对象的接口*/public OrderApi getProxyInterface(Order order) {// 设置被代理的对象,好方便invoke里面的操作this.order = order;// 把真正的订单对象和动态代理关联起来OrderApi orderApi = (OrderApi) Proxy.newProxyInstance(order.getClass().getClassLoader(), order.getClass().getInterfaces(), this);return orderApi;}/*** 实现invoke,在这个方法里面,具体判断当前是在调用什么方法,需要如何处理* @param proxy* @param method* @param args* @return* @throws Throwable*/@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//        System.out.println("用到的方法有什么"+method.getName());System.out.println("args有什么"+ Arrays.toString(args));if (method.getName().startsWith("set")) {System.out.println("进入setter方法了哟");// 如果不是创建人,那就不能修改if (order.getOrderUser() != null && order.getOrderUser().equals(args[1])) {// 可以操作return method.invoke(order,args);}else {
//                args[1] 表示第二个才是用户System.out.println("对不起, " + args[1] + ",您无权修改本订单中的数据");}}else if(method.getName().startsWith("to")){System.out.println("现在调用toString方法了哟");return method.invoke(order,args);}else {// 不是调用setter方法就继续执行return method.invoke(order,args);}return null;}}

Client

package day10代理模式;import day10代理模式.java中的代理类.DynamicProxy;public class Client {public static void main(String[] args) {// 1.张三先创建了一个订单Order order = new Order("设计模式", 100, "张三");// 2.创建一个动态代理DynamicProxy dynamicProxy = new DynamicProxy();// 3.将订单与动态代理关联起来OrderApi orderApi = dynamicProxy.getProxyInterface(order);//4.以下就需要使用被代理过的接口来操作了//4.1 李四想要来修改,那就会报错orderApi.setOrderNum(123,"李四");System.out.println("李四修改后的记录==>  "+orderApi);//4.2 张三修改orderApi.setOrderNum(123,"张三");System.out.println("张三修改后的记录==>  "+orderApi);}
}

解析

代理类首先是实现一个名为InvocationHandler的接口。需要持有被代理对象也就是OrderApi,然后通过一个对外提供的方法将代理与被代理绑定起来。实现invoke方法,这个方法就是为了判断订单接口中当前在使用的方法,对正在使用的方法做什么处理

一行一行解析就是首先判断方法名字是不是set开头的,如果是就看一下订单对象是否为空和这个方法传过来的值中的第1个下标对象是不是等于持有的对象,如果是就可以操作,如果不是就提示。

定义

 

代理模式调用示意图

代理模式的特点 

本质 

控制对象访问

何时选用 

 

 

http://www.lryc.cn/news/142156.html

相关文章:

  • vue2 路由进阶,VueCli 自定义创建项目
  • 《C语言编程环境搭建》工欲善其事 必先利其器
  • 蓝蓝设计ui设计公司作品案例-中节能现金流抗压测试软件交互及界面设计
  • 汽车制造业外发文件时 如何阻断泄密风险?
  • 怎么对App进行功能测试
  • 数字流的秩、单词频率(哈希实现)
  • 【洛谷】P2678 跳石头
  • Elasticsearch配置优化
  • Springboot整合minio组件-分布式文件存储
  • 多态/虚函数/虚函数表
  • QT中按钮的基类QAbstractButton
  • 并查集(种类并查集,带权并查集)
  • 飞天使-k8s基础组件分析-控制器
  • 有序充电运营管理平台是基于物联网和大数据技术的充电设施管理系统-安科瑞黄安南
  • LeetCode-227-基本计算器Ⅱ
  • dart 学习列表 List
  • 数据结构--树4.2.1(二叉树)
  • Presto之Driver个数
  • R语言响应面(RSM)、线性模型lm分析生产过程影响因素可视化
  • 剑指Offer --- 字符串篇
  • 7.elasticsearch同步工具-logstah
  • Redis之stream类型解读
  • C++ 网络编程项目fastDFS分布式文件系统(九)总结
  • 第五章 树与二叉树 一、树的定义与考点
  • C语言基础之——指针(下)
  • 小研究 - JVM 的类装载机制
  • 项目---日志系统
  • 设计模式--建造者模式(Builder Pattern)
  • 若依vue打印的简单方法
  • Rust 基础语法学习