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

【设计模式】设计模式之解释器模式

文章目录

  • 前言
  • 一、解释器模式
  • 二、最佳实践
  • 总结

前言

本节给大家讲一下设计模式中的解释器模式,并结合实际业务场景给大家讲解如何使用~

所有案例代码主要以Java语言为主, 好了, 废话不多说直接开整~

一、解释器模式

解释器模式(Interpreter Pattern)是一种行为型设计模式,它定义了一种语言的语法表示,并定义了一个解释器来解释这种语言中的表达式。

通过使用解释器模式,可以轻松地扩展和修改语言的语法,从而使其更加灵活。

该模式的结构包含以下组件:

  • 抽象表达式(AbstractExpression):定义了一个解释器中的抽象操作,每个具体的解释器都实现了该抽象操作。
  • 终结符表达式(TerminalExpression):定义了一个解释器中的终止操作,它不能再进行解释。
  • 非终结符表达式(NonterminalExpression):定义了一个解释器中的非终止操作,它可以递归调用其他解释器进行解释。
  • 环境(Context):保存了解释器需要的一些全局信息。

下面我们举一个大家日常开发中最常用的定时任务的例子:

// 抽象表达式
// 抽象表达式接口
public interface Expression {boolean interpret(String expression);
}// 秒表达式
public class SecondExpression implements Expression {@Overridepublic boolean interpret(String expression) {// 解析表达式,判断秒数是否匹配return true; // 返回是否匹配}
}// 分钟表达式
public class MinuteExpression implements Expression {@Overridepublic boolean interpret(String expression) {// 解析表达式,判断分钟数是否匹配return true; // 返回是否匹配}
}// 小时表达式
public class HourExpression implements Expression {@Overridepublic boolean interpret(String expression) {// 解析表达式,判断小时数是否匹配return true; // 返回是否匹配}
}// 日表达式
public class DayOfMonthExpression implements Expression {@Overridepublic boolean interpret(String expression) {// 解析表达式,判断日期是否匹配return true; // 返回是否匹配}
}// 定时任务表达式
public class CronExpression implements Expression {private Expression secondExpression;private Expression minuteExpression;private Expression hourExpression;private Expression dayOfMonthExpression;public CronExpression(Expression secondExpression, Expression minuteExpression,Expression hourExpression, Expression dayOfMonthExpression) {this.secondExpression = secondExpression;this.minuteExpression = minuteExpression;this.hourExpression = hourExpression;this.dayOfMonthExpression = dayOfMonthExpression;}@Overridepublic boolean interpret(String expression) {String[] fields = expression.split(" ");// 解析定时任务表达式,判断各个字段是否匹配boolean matchSecond = secondExpression.interpret(fields[0]);boolean matchMinute = minuteExpression.interpret(fields[1]);boolean matchHour = hourExpression.interpret(fields[2]);boolean matchDayOfMonth = dayOfMonthExpression.interpret(fields[3]);return matchSecond && matchMinute && matchHour && matchDayOfMonth;}
}public class Client {public static void main(String[] args) {// 创建表达式对象Expression second = new SecondExpression();Expression minute = new MinuteExpression();Expression hour = new HourExpression();Expression dayOfMonth = new DayOfMonthExpression();Expression cron = new CronExpression(second, minute, hour, dayOfMonth);// 判断定时任务是否应该被执行String expression = "0 0 1 * *"; // 每月1日执行任务boolean match = cron.interpret(expression);if (match) {System.out.println("定时任务执行中...");} else {System.out.println("定时任务未开始...");}}
}

当定时任务表达式为0 0 1 * *时,表示该定时任务在每个月的1日执行。如果当前日期是1日,那么match变量的值将为true,表示该定时任务应该被执行。否则,match变量的值将为false,表示该定时任务还未开始执行。表达式内部具体规则逻辑可以自行实现,这里只是为了方便演示~

二、最佳实践

在电商平台中,可以使用解释器模式来实现一个基于规则的折扣系统。该系统可以根据用户的购物车内容和优惠规则来计算出最终的折扣金额。

下面是一个示例,展示了如何使用解释器模式来实现一个基于规则的折扣系统:

// 抽象表达式
interface DiscountExpression {double interpret(ShoppingCart cart);
}// 终结符表达式:商品价格
class ItemPrice implements DiscountExpression {private String itemId;public ItemPrice(String itemId) {this.itemId = itemId;}public double interpret(ShoppingCart cart) {double price = 0;for (CartItem item : cart.getItems()) {if (item.getItemId().equals(itemId)) {price += item.getPrice() * item.getQuantity();}}return price;}
}// 终结符表达式:折扣价格
public class DiscountPrice implements  DiscountExpression {private double discount;public DiscountPrice(double discount) {this.discount = discount;}@Overridepublic double interpret(ShoppingCart cart) {double price = 0;for (CartItem item : cart.getItems()) {price += item.getPrice() * item.getQuantity() * discount;}return price;}
}// 终结符表达式:购物车总价
class CartTotal implements DiscountExpression {public double interpret(ShoppingCart cart) {double total = 0;for (CartItem item : cart.getItems()) {total += item.getPrice() * item.getQuantity();}return total;}
}// 非终结符表达式:满减
class Discount implements DiscountExpression {private DiscountExpression condition;private DiscountExpression action;public Discount(DiscountExpression condition, DiscountExpression action) {this.condition = condition;this.action = action;}public double interpret(ShoppingCart cart) {if (condition.interpret(cart) > 0) {return action.interpret(cart);}return 0;}
}// 环境:购物车
class ShoppingCart {private List<CartItem> items;public ShoppingCart() {items = new ArrayList<>();}public void addItem(CartItem item) {items.add(item);}public List<CartItem> getItems() {return items;}
}// 环境:购物车商品项
class CartItem {private String itemId;private String itemName;private double price;private int quantity;public CartItem(String itemId, String itemName, double price, int quantity) {this.itemId = itemId;this.itemName = itemName;this.price = price;this.quantity = quantity;}public String getItemId() {return itemId;}public String getItemName() {return itemName;}public double getPrice() {return price;}public int getQuantity() {return quantity;}
}// 客户端
public class DiscountSystemDemo {public static void main(String[] args) {// 初始化购物车ShoppingCart cart = new ShoppingCart();cart.addItem(new CartItem("001", "商品1", 100, 2));cart.addItem(new CartItem("002", "商品2", 200, 1));// 定义折扣规则DiscountExpression rule1 = new Discount(new ItemPrice("001"), new DiscountPrice(0.1));DiscountExpression rule2 = new Discount(new ItemPrice("002"), new DiscountPrice(0.2));// 计算折扣double discount1 = rule1.interpret(cart);double discount2 = rule2.interpret(cart);double totalDiscount = discount1 + discount2;// 输出结果System.out.println("折扣总额:" + totalDiscount);System.out.println("应付金额:" + (new CartTotal().interpret(cart) - totalDiscount));//        折扣总额:120.0
//        应付金额:280.0}
}

上述仅仅是一个简单的示例,实际应用中可能会更加复杂。例如,可能需要实现更多的折扣规则类型,或者添加更多的操作和函数。但是,无论怎样扩展和修改折扣系统的规则,解释器模式都可以帮助我们轻松地完成这些任务。

总结

解释器模式其实并不难,大家在学习的时候一定要在理解的基础上去写代码,不要去背代码。

相关文章:

【设计模式】设计模式之解释器模式

文章目录 前言一、解释器模式二、最佳实践总结 前言 本节给大家讲一下设计模式中的解释器模式,并结合实际业务场景给大家讲解如何使用~ 所有案例代码主要以Java语言为主, 好了, 废话不多说直接开整~ 一、解释器模式 解释器模式&#xff08;Interpreter Pattern&#xff09;…...

leetcode551. 学生出勤记录 I

题目描述解题思路执行结果 leetcode551. 学生出勤记录 I . 题目描述 给你一个字符串 s 表示一个学生的出勤记录&#xff0c;其中的每个字符用来标记当天的出勤情况&#xff08;缺勤、迟到、到场&#xff09;。记录中只含下面三种字符&#xff1a; A&#xff1a;Absent&#xff…...

flume拦截器介绍

Flume是一个开源的、分布式的、可靠的、高效的海量数据采集、聚合和传输系统。其中&#xff0c;拦截器(Interceptor)是Flume中的一种组件&#xff0c;可以在数据(Event)流动的各个阶段对数据进行处理、过滤或转换&#xff0c;从而实现更为灵活、高效的数据采集和传输。 1、拦截…...

5.4、服务器编程基本框架和两种高效的事件处理模式

5.4、服务器编程基本框架和两种高效的事件处理模式 1.服务器编程基本框架2.两种高效的事件处理模式①Reactor模式②Proactor模式③模拟Proactor模式 1.服务器编程基本框架 模块功能I/O 处理单元处理客户连接&#xff0c;读写网络数据逻辑单元业务进程或线程网络存储单元数据库、…...

Flink主要有两种基础类型的状态:operator state。

Flink主要有两种基础类型的状态&#xff1a;keyed state 和operator state。 Operator State 对于Operator State(或者non-keyed state)&#xff0c;每个operator state绑定到一个并行operator实例上。在Flink中&#xff0c;Kafka Connector是一个使用Operator State的很好的例…...

【vue2】使用vue-admin-template动态添加路由的思路/addRoutes的使用

&#x1f609;博主&#xff1a;初映CY的前说(前端领域) ,&#x1f4d2;本文核心&#xff1a;用原生js实现省市区联动 【前言】在通用的后台管理项目的开发中&#xff0c;不仅仅是会涉及到对表单数据等的增删改查操作还会涉及到一些关于权限管理的问题。我们将基于一个RBAC的思维…...

Python语言中的注释方法应用

Python语言中的注释方法 在Python编程中&#xff0c;与其他编程语言一样&#xff0c;有良好的注释部分&#xff0c;会让你的程序在后续的改进或优化中&#xff0c;变得便利。同时&#xff0c;给自己培养了良好的编程习惯。 在Python语言中&#xff0c;有两种注释方法。 1.单行…...

Google浏览器翻译无法正常使用解决

1.查找可用服务器地址 按WinR键打开运行→输入cmd回车&#xff0c;打开命令提示符→输入ping google.cn 回车。记录一下下图红框里的ip地址&#xff0c;一会要用到 最近自己ping出来的ip可能不能用了&#xff0c;可以尝试用下面的ip 142.251.163.90 142.250.113.90 142.251.…...

ETCD(三)操作指令

1. put put #将给定的key写入到存储 --ignore-lease[false] #使用当前租约更新key --ignore-value[false] #使用当前值更新key --lease"0" # 要附加到key的租约ID&#xff08;十六进制&#xff09; --prev-kv[false] # 返回修改前的上一个键值对2. get get #获取给…...

小白学Pytorch系列--Torch.optim API Base class(1)

小白学Pytorch系列–Torch.optim API Base class(1) torch.optim是一个实现各种优化算法的包。大多数常用的方法都已得到支持&#xff0c;而且接口足够通用&#xff0c;因此将来还可以轻松集成更复杂的方法。 如何使用优化器 使用手torch.optim您必须构造一个优化器对象&…...

flac格式如何转mp3,3招帮你搞定

flac格式如何转mp3&#xff0c;3招帮你搞定的方法来啦。当你的音频是flac格式是不是很头疼&#xff0c;又不知道怎么转mp3 。然后网上搜索出很多方法又不知道从哪个下手&#xff0c;是不是很疑惑&#xff1f;那今天就来看看小编推荐的方法吧&#xff0c;一定让你眼前一亮&#…...

Redis入门到入土(day01)

NoSQL概述 为什么用NoSQL 1、单机MySQL的美好年代 在90年代&#xff0c;一个网站的访问量一般不大&#xff0c;用单个数据库完全可以轻松应付&#xff01; 在那个时候&#xff0c;更多的都是静态网页&#xff0c;动态交互类型的网站不多。 上述架构下&#xff0c;我们来看看…...

JVM垃圾回收GC 详解(java1.8)

目录 垃圾判断算法&#xff08;你是不是垃圾&#xff1f;&#xff09; 引用计数法 可达性算法 对象的引用 强引用 软引用 弱引用 虚引用 对象的自我救赎 垃圾回收算法--分代 标记清除算法 复制算法 标记整理法 垃圾处理器 垃圾判断算法&#xff08;你是不是垃圾&…...

Mybatis-Plus -03 Mybatis-Plus实现CRUD

Mybatis-Plus实现CRUD 1 Insert增加2 ID生成策略3 Delete删除4 逻辑删除5 Update修改6 Select查询 Mybatis-Plus实现CRUD 通用 CRUD 封装**BaseMapper (opens new window)**接口&#xff0c;为 Mybatis-Plus 启动时自动解析实体表关系映射转换为 Mybatis 内部对象注入容器参数 …...

综合能源系统中基于电转气和碳捕集系统的热电联产建模与优化研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…...

“智慧赋能 强链塑链”|工程物资供应链管理中的数字化应用

工程项目中的供应链管理至关重要 工程建设行业是国民经济的重要支柱之一&#xff0c;虽然在总产值上持续保持增长态势&#xff0c;但近年来行业的利润总额增速已连续多年呈现下降趋势。究其原因&#xff0c;可以大体从两个方面来看&#xff1a;一是行业盈利能力出现下降&#x…...

通过docker发布项目

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言例如&#xff1a;docker项目的发布方式 [docker发布的参考链接](https://www.cnblogs.com/emperorking/articles/11244253.html) 一、docker是什么&#xff1f;…...

为什么Spring和IDEA不推荐使用@Autowired注解?

在Spring开发中&#xff0c;Autowired注解是一个常用的依赖注入方式。但是&#xff0c;你可能会惊奇地发现&#xff0c;Spring和IDEA都不推荐使用Autowired注解。关于这个问题&#xff0c;其实答案相对统一&#xff0c;实际上用大白话说起来也容易理解。 官方答案 首先&#…...

windows下运行dpdk下的helloworld

打开“本地安全策略”管理单元,在搜索框输入secpol。 打开本地策略->用户权限分配->锁定内存页->添加用户或组->高级->立即查找 输入电脑用户名,选择并添加。点击确定后,重启电脑。 安装内核驱动,下载地址https://download.csdn.net/download/qq_36314864…...

【AI理论学习】深入理解Prompt Learning和Prompt Tuning

深入理解Prompt Learning和Prompt Tuning 背景Prompt Learning简介1. Prompt是什么&#xff1f;2. 为什么要使用Prompt&#xff1f;3. Prompt Learning的形式&#xff08;举例&#xff09;4. 有哪些Pre-training language model&#xff1f;5. 常见的Prompt Learning的方法 Pro…...

变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析

一、变量声明设计&#xff1a;let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性&#xff0c;这种设计体现了语言的核心哲学。以下是深度解析&#xff1a; 1.1 设计理念剖析 安全优先原则&#xff1a;默认不可变强制开发者明确声明意图 let x 5; …...

OpenLayers 可视化之热力图

注&#xff1a;当前使用的是 ol 5.3.0 版本&#xff0c;天地图使用的key请到天地图官网申请&#xff0c;并替换为自己的key 热力图&#xff08;Heatmap&#xff09;又叫热点图&#xff0c;是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

从零实现富文本编辑器#5-编辑器选区模型的状态结构表达

先前我们总结了浏览器选区模型的交互策略&#xff0c;并且实现了基本的选区操作&#xff0c;还调研了自绘选区的实现。那么相对的&#xff0c;我们还需要设计编辑器的选区表达&#xff0c;也可以称为模型选区。编辑器中应用变更时的操作范围&#xff0c;就是以模型选区为基准来…...

Oracle查询表空间大小

1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)

概述 在 Swift 开发语言中&#xff0c;各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过&#xff0c;在涉及到多个子类派生于基类进行多态模拟的场景下&#xff0c;…...

【论文笔记】若干矿井粉尘检测算法概述

总的来说&#xff0c;传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度&#xff0c;通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...

Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?

在大数据处理领域&#xff0c;Hive 作为 Hadoop 生态中重要的数据仓库工具&#xff0c;其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式&#xff0c;很多开发者常常陷入选择困境。本文将从底…...

《C++ 模板》

目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板&#xff0c;就像一个模具&#xff0c;里面可以将不同类型的材料做成一个形状&#xff0c;其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式&#xff1a;templa…...

保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek

文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama&#xff08;有网络的电脑&#xff09;2.2.3 安装Ollama&#xff08;无网络的电脑&#xff09;2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...