【设计模式-4.11】行为型——解释器模式

news2025/6/7 15:18:49

说明:本文介绍行为型设计模式之一的解释器模式

定义

解释器模式(Interpreter Pattern)指给定一门语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子。解释器模式是一种按照规定的文法(语法)进行解析的模式,属于行为型设计模式。

(引自《设计模式就该这样学》P385)

自定义脚本

这里介绍《秒懂设计模式》这本书中的举例,假设我们需要定义一个刷怪脚本,脚本语言如下,执行顺序为自上而下。

BEGIN					// 脚本开始
MOVE 500,600			// 鼠标移动到屏幕(500, 600)的位置
	BEGIN LOOP 5		// 开始循环5次
		LEFT_CLICK;		// 循环鼠标单击左键
		DELAY 1;		// 每次延迟1秒
	END;				// 循环结束
RIGHt_LEFT;				// 按下鼠标左键
DELAY 7200;				// 延迟7200秒
END:					// 脚本结束

分析一下,这个脚本中包含了一下操作:

  • 移动鼠标

  • 鼠标左键单击(包含按下鼠标左键、松开鼠标左键)

  • 按下鼠标左键

  • 延迟

也就是五个操作:移动鼠标、鼠标左键按下、鼠标左键松开、鼠标左键单击(由鼠标左键按下、鼠标左键松开组合而成)、延迟

另外,还包含了脚本执行的语法顺序,如下:

  • 循环:循环执行某些操作

  • 顺序:脚本的执行是从上到下,顺序执行的


分析完,将上面的脚本语法转为代码语言,如下:

(表达式接口,Expression)

/**
 * 表达式接口
 */
public interface Expression {

    /**
     * 解释
     */
    void interpret();
}

以下都是解释器,对应上述分析的操作和流程

(移动鼠标,MoveMouse)

/**
 * 移动鼠标
 */
public class MoveMouse implements Expression {

    /**
     * x坐标
     */
    private int x;

    /**
     * y坐标
     */
    private int y;


    public MoveMouse(int x, int y) {
        this.x = x;
        this.y = y;
    }

    @Override
    public void interpret() {
        System.out.println("移动鼠标到【" + x + "," + y + "】的位置");
    }
}

(鼠标左键按下,LeftKeyDown)

/**
 * 鼠标左键按下
 */
public class LeftKeyDown implements Expression {

    @Override
    public void interpret() {
        System.out.println("按下鼠标左键");
    }
}

(鼠标左键松开,LeftKeyUp)

/**
 * 鼠标左键松开
 */
public class LeftKeyUp implements Expression {

    @Override
    public void interpret() {
        System.out.println("松开鼠标左键");
    }
}

(鼠标左键单击,LeftKeyClick)

/**
 * 鼠标左键单击
 */
public class LeftKeyClick implements Expression {

    /**
     * 左键按下
     */
    private Expression leftKeyDown;

    /**
     * 左键松开
     */
    private Expression leftKeyUp;

    /**
     * 左键按下
     */
    public LeftKeyClick() {
        this.leftKeyDown = new LeftKeyDown();
        this.leftKeyUp = new LeftKeyUp();
    }

    /**
     * 单击鼠标左键就是先按下鼠标左键, 再松开鼠标左键
     */
    @Override
    public void interpret() {
        leftKeyDown.interpret();
        leftKeyUp.interpret();
    }
}

(延迟,Delay)

/**
 * 延迟解释器
 */
public class Delay implements Expression {

    /**
     * 延迟秒数
     */
    private int seconds;

    public Delay(int seconds) {
        this.seconds = seconds;
    }

    public int getSeconds() {
        return seconds;
    }

    @Override
    public void interpret() {
        System.out.println("系统延迟:" + seconds + "秒");
        try {
            Thread.sleep(seconds * 1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}

(循环,Repetition)

/**
 * 循环表达式解释器
 */
public class Repetition implements Expression {

    /**
     * 循环次数
     */
    private int loopCount;

    /**
     * 循环体内的表达式
     */
    private Expression loopBodySequence;

    public Repetition(Expression loopBodySequence, int loopCount) {
        this.loopBodySequence = loopBodySequence;
        this.loopCount = loopCount;
    }

    @Override
    public void interpret() {
        while (loopCount > 0) {
            loopBodySequence.interpret();
            loopCount--;
        }
    }
}

(顺序,Sequence)

import java.util.List;

/**
 * 顺序
 */
public class Sequence implements Expression {

    /**
     * 脚本包含了多个表达式
     */
    private List<Expression> expressions;

    public Sequence(List<Expression> expressions) {
        this.expressions = expressions;
    }

    /**
     * 顺序执行表达式
     */
    @Override
    public void interpret() {
        for (Expression expression : expressions) {
            expression.interpret();
        }
    }
}

客户端使用,将开头的脚本按照规则放入,并解释(为了节约时间,将最后的延迟时间换成10秒)

import java.util.Arrays;

public class Client {
    public static void main(String[] args) {
        new Sequence(Arrays.asList(
                new MoveMouse(500, 600),
                new Repetition(
                        new Sequence(
                                Arrays.asList(
                                        new LeftKeyClick(), new Delay(1)
                                )
                        ), 5
                ),
                new LeftKeyDown(),
                new Delay(10)
        )).interpret();
    }
}

执行,可见每一个表达式都被成功解释

在这里插入图片描述

代码看下来,我认为解释器模式结构上很简单,就是定义一个接口,生成多个解释器实现类,具体的实现逻辑需要根据实际的业务实现。

使用场景

在《设计模式就该这样学》(P386)这本书中,提到解释器模式适用于以下场景:

(1)一些重复出现的问题可以用一种简单的语言进行表示。

(2)一个简单语法需要解释的场景。

例如,项目中需要解析cron表达式,根据解析后的内容去做相应的业务,我想可以考虑使用解释器模式。

总结

本文介绍了行为型设计模式中的访问者模式,参考《设计模式就该这样学》、《秒懂设计模式》两书,自定义脚本是《秒懂设计模式》中的举例。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2401361.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【已解决】MACOS M4 芯片使用 Docker Desktop 工具安装 MICROSOFT SQL SERVER

1. 环境准备 确认 Docker Desktop 配置 确保已安装 Docker Desktop for Mac (Apple Silicon)&#xff08;版本 ≥ 4.15.0&#xff09;。开启 Rosetta&#xff08;默认开启&#xff09;&#xff1a; 打开 Docker Desktop → Settings → General → Virtual Machine Options …

Quipus系统的视频知识库的构建原理及使用

1 原理 VideoRag在LightRag基础上增加了对视频的处理&#xff0c;详细的分析参考LightRag的兄弟项目VideoRag系统分析-CSDN博客。 Quipus的底层的知识库的构建的核心流程与LightRag类似&#xff0c;但在技术栈的选择和处理有所不同。Quipus对于视频的处理实现&#xff0c;与Vi…

web3-去中心化金融深度剖析:DEX、AMM及兑换交易传播如何改变世界

web3-去中心化金融深度剖析&#xff1a;DEX、AMM及兑换交易传播如何改变世界 金融问题 1.个人投资&#xff1a;在不同的时间和可能的情况&#xff08;状态&#xff09;下积累财富 2.商业投资&#xff1a;为企业家和企业提供投资生产性活动的资源 目标&#xff1a;跨越时间和…

国芯思辰|SCS5501/5502芯片组打破技术壁垒,重构车载视频传输链路,兼容MAX9295A/MAX96717

在新能源汽车产业高速发展的背景下&#xff0c;电机控制、智能驾驶等系统对高精度信号处理与高速数据传输的需求持续攀升。 针对车载多摄像头与自动驾驶辅助系统对长距离、低误码率、高抗干扰性数据传输的需求&#xff0c;SCS5501串行器与SCS5502解串器芯片组充分利用了MIPI A…

压敏电阻的选型都要考虑哪些因素?同时注意事项都有哪些?

压敏电阻&#xff0c;英文名简称VDR&#xff0c;电子元器件中重要的成员之一&#xff0c;是一种非线性伏安特性的电阻器件&#xff0c;有电阻特性的同时&#xff0c;也拥有其他自身的特性&#xff0c;广泛应用于众多领域。在电源系统、安防系统、浪涌抑制器、电动机保护、汽车电…

用WPDRRC模型,构建企业安全防线

文章目录 前言什么是 WPDRRC 模型预警&#xff08;Warning&#xff09;保护&#xff08;Protection&#xff09;检测&#xff08;Detection&#xff09;响应&#xff08;Response&#xff09;恢复&#xff08;Recovery&#xff09;反击&#xff08;Counterattack&#xff09; W…

使用 Amazon Q Developer CLI 快速搭建各种场景的 Flink 数据同步管道

在 AI 和大数据时代&#xff0c;企业通常需要构建各种数据同步管道。例如&#xff0c;实时数仓实现从数据库到数据仓库或者数据湖的实时复制&#xff0c;为业务部门和决策团队分析提供数据结果和见解&#xff1b;再比如&#xff0c;NoSQL 游戏玩家数据&#xff0c;需要转换为 S…

Linux 里 su 和 sudo 命令这两个有什么不一样?

《小菜狗 Linux 操作系统快速入门笔记》目录&#xff1a; 《小菜狗 Linux 操作系统快速入门笔记》&#xff08;01.0&#xff09;文章导航目录【实时更新】 Linux 是一个多用户的操作系统。在 Linux 中&#xff0c;理论上来说&#xff0c;我们可以创建无数个用户&#xff0c;但…

JAVASCRIPT 简化版数据库--智能编程——仙盟创梦IDE

// 数据模型class 仙盟创梦数据DM {constructor(key) {this.key ${STORAGE_PREFIX}${key};this.data this.加载数据();}加载数据() {return JSON.parse(localStorage.getItem(this.key)) || [];}保存() {localStorage.setItem(this.key, JSON.stringify(this.data));}新增(it…

命名管道实现本地通信

目录 命名管道实现通信 命名管道通信头文件 创建命名管道mkfifo 删除命名管道unlink 构造函数 以读方式打开命名管道 以写方式打开命名管道 读操作 写操作 析构函数 服务端 客户端 运行结果 命名管道实现通信 命名管道通信头文件 #pragma#include <iostream> #include &l…

iOS上传应用包错误问题 “Invalid bundle. The “UIInterfaceOrientationPortrait”“

引言 在开发 iOS 应用的整个生命周期中&#xff0c;打包上传到 App Store 是一个至关重要的步骤。每一次提交&#xff0c;Xcode 都会在后台执行一系列严格的校验流程&#xff0c;包括对 Info.plist 配置的检查、架构兼容性的验证、资源完整性的审查等。如果某些关键项配置不当…

猎板厚铜PCB工艺能力如何?

在电子产业向高功率、高集成化狂奔的今天&#xff0c;电路板早已不是沉默的配角。当5G基站、新能源汽车、工业电源等领域对电流承载、散热效率提出严苛要求时&#xff0c;一块能够“扛得住大电流、耐得住高温”的厚铜PCB&#xff0c;正成为决定产品性能的关键拼图。而在这条赛道…

Flutter快速上手,入门教程

目录 一、参考文档 二、准备工作 下载Flutter SDK&#xff1a; 配置环境 解决环境报错 zsh:command not found:flutter 执行【flutter doctor】测试效果 安装Xcode IOS环境 需要安装brew&#xff0c;通过brew安装CocoaPods. 复制命令行&#xff0c;打开终端 分别执行…

算法:前缀和

1.【模版】前缀和 【模板】前缀和_牛客题霸_牛客网 这道题如果使用暴力解法时间复杂度为O(n*m)&#xff0c;会超时&#xff0c;所以要使用前缀和算法。 前缀和->快速求出数组中某一个连续区间的和。 第一步&#xff1a;预处理出一个前缀和数组 dp。 dp[i]表示[1, i] 区间…

DEVICENET转MODBUS TCP网关与AB数据输出模块的高效融合方案研究

在工业自动化领域&#xff0c;多样化的设备通常采用不同的通信协议&#xff0c;这为系统集成带来了显著的挑战。特别是在需要将遵循DeviceNet协议的设备与基于MODBUS TCP协议的系统进行互连时&#xff0c;这一挑战尤为突出。AB数据输出作为一种功能卓越的DeviceNet分布式输入/输…

牛客小白月赛113

前言&#xff1a;这场的E题补的我头皮都发麻了。 A. 2025 题目大意&#xff1a;一个仅有‘-’‘*’组成的字符串&#xff0c;初始有一个sum 1&#xff0c; 从左到右依次遍历字符串&#xff0c;遇到-就让sum--&#xff1b;遇到*就让sum* 2&#xff0c;问sum有没有可能大于等于…

Mac版本Android Studio配置LeetCode插件

第一步&#xff1a;Android Studio里面找到Settings&#xff0c;找到Plugins&#xff0c;在Marketplace里面搜索LeetCode Editor。 第二步&#xff1a;安装对应插件&#xff0c;并在Tools->LeetCode Plugin页面输入帐号和密码。 理论上&#xff0c;应该就可以使用了。但是&a…

电子电路基础1(杂乱)

电路基础知识 注意&#xff1a;电压源与电流源的表现形式 注意&#xff1a;在同一根导线上电势相等 电阻电路的等效变换 电子元器件基础 电阻

【openssl】升级为3.3.1,避免安全漏洞

本文档旨在形成 对Linux系统openssl版本进行升级 的搭建标准操作过程&#xff0c;搭建完成后&#xff0c;实现 openssl 达到3.3以上版本&#xff0c;避免安全漏洞 效果。 一、查看当前版本 版本不高于3.1的&#xff0c;均需要升级。 # 服务器上运行以下命令&#xff0c;查看…

使用 HTML +JavaScript 从零构建视频帧提取器

在视频编辑、内容分析和多媒体处理领域&#xff0c;常常需要从视频中提取关键帧。手动截取不仅效率低下&#xff0c;还容易遗漏重要画面。本文介绍的视频帧提取工具通过 HTML5 技术栈实现了一个完整的浏览器端解决方案&#xff0c;用户可以轻松选择视频文件并进行手动或自动帧捕…