22行为型设计模式——解释器模式

news2024/9/22 3:48:33

一、解释器模式

解释器模式(Interpreter Pattern)是一种行为型设计模式,主要用于解析和解释特定的语言或表达式。它的核心思想是为语言中的每种语法规则定义一个解释器,通过这些解释器将语言的表示形式转换为可执行的操作。解释器模式在实际业务中使用频率相对较低,因为其性能和类膨胀的问题,需求较少。在处理解析和解释特定的语言或类似表达式转换或者算术表达式计算等复杂计算逻辑的场景,这种模式常见。

解释器模式的结构图

  1. 抽象表达式(Abstract Expression)角色:定义解释器的接口,约定解释器的解释操作。

  2. 终结符表达式(Terminal Expression)角色:实现文法中与终结符相关的操作。

  3. 非终结符表达式(Nonterminal Expression)角色:实现文法中与非终结符相关的操作。

  4. 环境(Context)角色:包含解释器需要的数据或是公共的功能。

  5. 客户端(Client):主要任务是将需要分析的句子或表达式转换成抽象语法树,然后调用解释器的解释方法。

解释器模式涉及三个关键的元素

  1. 抽象语法树:在解释器模式中,语言的句子通常表示为抽象语法树(AST)。AST是句子的树形表示,它反映了句子的结构,使得解释器可以按照树的结构来解释句子。

  2. 文法与句子:文法是用于描述语言的语法结构的形式规则。句子是语言的基本单位,由终结符构成,能由“文法”推导出。

  3. 解释器:解释器是使用解释器模式设计的类,它用于解释和执行语言的句子。

二、解释器模式的设计方法

设计代码实现一个简单的算术表达式解释器,能够处理加法和减法操作,并且可以从终端读取表达式进行求值。终端输入表达式格式:[数字][空格][操作符][空格][数字]

interpreter.cpp

#include <iostream>
#include <sstream>
#include <string>
#include <vector>
#include <stack>
#include <cctype>
#include <stdexcept>

// 接口
class Expression {
public:
	virtual ~Expression() {}
	virtual int interpreter() const = 0;
};

// 数值表达式
class NumberExpression : public Expression {
public:
	NumberExpression(int value) : value_(value) {}
	
	int interpreter() const override {
		return value_;
	}
	
private:
	int value_;
};

// 加法表达式
class AdditionExpression : public Expression {
public:
	AdditionExpression(Expression* left, Expression* right) : left_(left), right_(right) {}
	
	int interpreter() const override {
		return left_->interpreter() + right_->interpreter();
	}
	
	~AdditionExpression() {
		delete left_;
		delete right_;
	}
	
private:
	Expression* left_;
	Expression* right_;
};

// 减法表达式
class SubtractionExpression : public Expression {
public:
	SubtractionExpression(Expression* left, Expression* right) : left_(left), right_(right) {}
	
	int interpreter() const override {
		return left_->interpreter() - right_->interpreter();	
	}
	
	~SubtractionExpression() {
		delete left_;
		delete right_;
	}
private:
	Expression* left_;
	Expression* right_;
};

class ExpressionEvaluator {
public:
	static int evaluate(const std::vector<std::string>& tokens) {
		std::stack<Expression*> values;
		std::stack<std::string> operators;
		
		for (const auto& token : tokens) {
		if (isdigit(token[0])) {
			values.push(new NumberExpression(std::stoi(token)));
		} else if (token == "+" || token == "-") {
			while (!operators.empty() && (precedence(operators.top()) >= precedence(token))) {
				processOPerator(values,operators);
			}
			operators.push(token);
		} else {
			throw std::runtime_error("Syntax error: 无效的符号 " + token);
		}
	}
	
	while (!operators.empty()) {
		processOPerator(values,operators);
	}
	
	if (values.size() != 1) {
		throw std::runtime_error("Syntax error: 无效的算术表达式");
	}
	
	int  result = values.top()->interpreter();
	delete values.top();
	return result;
	}

private:
	static int precedence(const std::string& operate) {
		if (operate == "+" || operate == "-") return 1;
		return 0; 
	}
	
	static void processOPerator(std::stack<Expression*>& values, std::stack<std::string>&operators) {
		if (values.size() <2) {
			throw std::runtime_error("Syntax error: 操作数不足");
		}
		
		Expression* right = values.top(); values.pop();
		Expression* left = values.top(); values.pop();
		std::string operate = operators.top(); operators.pop();
		
		Expression* result = nullptr;
		if (operate == "+") {
		    result = new AdditionExpression(left, right);
		} else if (operate == "-") {
		    result = new SubtractionExpression(left, right);
		} else {
		    throw std::runtime_error("Syntax error: 无效的操作符 " + operate);
		}

        	values.push(result);
	}
};

// 客户端
int main() {
	std::cout << "\n// 示例用法1 " << std::endl;
	std::cout << "终端输入 '3 + 4 - 2' " << std::endl;
	std::vector<std::string> tokens = {"3", "+", "4", "-", "2"};
	try {
		int result = ExpressionEvaluator::evaluate(tokens);
		std::cout << "算术表达式结果: " << result << std::endl;
	} catch (const std::exception& e) {
		std::cerr << "Error: " << e.what() << std::endl;
	}
	
	std::cout << "\n// 示例用法2 " << std::endl;
	std::cout << "终端输入 '10 + 20 - 30 + 40' " << std::endl;
	tokens = {"10", "+", "20", "-", "30", "+", "40"};
	try {
		int result = ExpressionEvaluator::evaluate(tokens);
		std::cout << "算术表达式结果: " << result << std::endl;
	} catch (const std::exception& e) {
		std::cerr << "Error: " << e.what() << std::endl;
	}
	
	tokens.clear();
	
	std::string line;
	std::cout <<  "\n请输入算术表达式 (例如:'3 + 4 - 2'): ";
	std::getline(std::cin, line);
	
	std::istringstream iss(line);
	std::string token;
	
	while (iss >> token) {
		if (token != "+" && token != "-" && !isdigit(token[0])) {
			std::cerr << "Error: 无效的符号 " << token << std::endl;
            		return 1;
		}
		tokens.push_back(token);
	}
	
	try {
		int result = ExpressionEvaluator::evaluate(tokens);
		std::cout << "算术表达式结果: " << result << std::endl;
	} catch (const std::exception& e) {
		std::cerr << "Error: " << e.what() << std::endl;
		return 1;
	}
	
	return 0;
}

代码中使用了中缀表示法(即常见的 3 - 4 + 5 这种形式),并且利用了栈来处理操作符和操作数。需要注意的是数字包括符号之间需要有空格。(空格作为切割符)

运行效果

 

三、解释器模式的应用场景

  1. 编程语言的解释器:解释器模式常用于构建编程语言的解释器。例如,CPython、V8,你也可以设计一个自定义的小型编程语言,通过定义解释器模式中的表达式和语法规则来解析和执行这些语言。

  2. 复杂的计算器:当你需要构建一个可以解析并计算复杂数学表达式的计算器时,解释器模式可以帮助将数学表达式分解为更简单的操作,然后逐步计算结果,例如上面的代码实现中interpreter.cpp。

  3. 规则引擎:在某些应用场景中,需要根据特定的规则或条件执行不同的操作。解释器模式可以用来实现这些规则的解析和执行,例如业务规则引擎。

  4. DSL(领域特定语言):如果你需要为特定领域创建一个定制的语言或语法(例如用于配置或查询),解释器模式可以帮助你解析这种领域特定语言的表达式,并执行相应的操作。

  5. 数据处理和转换:在需要将输入数据转换为特定格式的场景中,解释器模式可以帮助将数据解析成中间表示,然后进行转换和处理。

  6. 脚本语言:当需要实现一个小型脚本语言,或在应用中嵌入一个脚本解释器时,解释器模式提供了一种组织和解释脚本语言的方式。

四、总结

解释器模式允许你可以将复杂的语句表达式转换成对应的执行代码。这种模式通常用于字符串解析或语言解释器中。它将算术或逻辑表达式设计成一个接一个的独立的“组件”,每个组件负责解释其中一个语法符号的含义。这些组件可以单独构建、组合或重用。

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

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

相关文章

240831-RAG新利器之Kotaemon的安装与配置

A. 用户界面 该项目既可以作为功能性 RAG UI&#xff0c;既可以用于对文档进行 QA 的最终用户&#xff0c;也可以用作想要构建自己的 RAG 管道的开发人员。对于最终用户&#xff1a; - 一个干净且简约的用户界面&#xff0c;用于基于RAG的QA。 - 支持 LLM API 提供程序&#xf…

最小栈

最小栈 这题难就难在要能在常数时间内检索到最小元素。 单纯用一个变量记录最小值是无法实现常数时间内获取最小元素的&#xff0c;这个时候我们根据栈的特性&#xff0c;另开一个辅助栈&#xff0c;存储我们的栈里每个时刻的最小值。代码&#xff1a; class MinStack {stac…

idea的全局配置

这样一来&#xff0c;每次创建新项目完就不用每次改配置了

内存管理篇-16二级页表工作原理

1.修正上节课的转换图 上节课的页表的一级页表其实并不完全正确&#xff0c;一般虚拟页帧和物理页帧号不会都占用实际字段&#xff0c;这样毕竟很浪费内存。 2.再分析一下页表的开销情况&#xff1a; 一级页表&#xff1a;以4KB物理页为映射单位&#xff0c;每个进程4MB的虚…

【Python】家庭用电数据的时序分析

Household Electricity Consumption | Kaggle 目录 数据简介 探索分析 数据清洗 用电占比 趋势分析 序列分解 周期分析 周期分解 分析小结 数据简介 240000-household-electricity-consumption-records数据集包含了一个家庭6个月的用电数据&#xff0c;收集于2007年1…

HTB-Appointment(SQL注入-万能钥匙)

前言 各位师傅大家好&#xff0c;我是qmx_07&#xff0c;今天给大家讲解Appointment靶场&#xff0c;这一章节 我们涉及到一些web安全的相关知识 渗透过程 信息搜集 我们通过端口扫描&#xff0c;得知对方开启了http80端口思路&#xff1a;尝试扫描下敏感目录 目录扫描(gob…

PTH哈希传递攻击

PTH哈希传递攻击&#xff08;Pass The Hash&#xff09; 一、PTH简介 1、攻击原理 在使用 NTLM 身份验证的系统或服务上&#xff0c;用户密码永远不会以明文形式通过网络发送。 Windows 上的应用程序要求用户提供明文密码&#xff0c;然后调用 LsaLogonUser 类的 API&#x…

AR 眼镜之-系统通知定制(通知中心)-实现方案

目录 &#x1f4c2; 前言 AR 眼镜系统版本 系统通知定制 1. &#x1f531; 技术方案 1.1 技术方案概述 1.2 实现方案 1&#xff09;通知弹窗消失 2&#xff09;通知中心显示 2. &#x1f4a0; 通知弹窗消失 2.1 通知弹窗显示时长到期后自动消失 2.2 将通知添加到通知…

vue3中ref绑定的节点顺序错乱

问题复现 眨眼睛这个是修正过了的&#xff0c;小友的应该是ref直接绑定navigationTextList对吧&#xff0c; 按正常想法肯定是既然这个数组会动态更新&#xff0c;我只需要index不就能确定是哪个节点啦&#xff0c;倘若只是静态数据应该不会有什么问题&#xff0c; ⚠️但如果出…

想了个创业的点子问老婆,她说你这不就是外包公司吗

年近35&#xff0c;老在想着万一毕业了咋整&#xff0c;其他的技能也不会&#xff0c;只能去“吉祥三保”、“铁人三项”了&#xff0c;但是吧又不甘心这十来年的“手艺”&#xff0c;又想着这几年大环境下那么多失业的同行&#xff0c;是不是也都有这方面的需求&#xff0c;于…

#读书#经济#《宏观经济学》by N.Gregory Mankiw 第十版 - 第2篇 古典理论:长期中的经济 - 第6章 开放的经济 - 6.3 汇率

在前面两节讨论了产品与服务以及资本在国家之间的流动之后&#xff0c;本节深入讨论了进行这些交易的价格&#xff0c;即两个国家之间进行贸易往来时使用的汇率。从介绍实际汇率和名义汇率的概念开始&#xff0c;讨论实际汇率和贸易余额的关系、两种汇率的决定因素以及财政政策…

九、JMeter之压力测试

文章目录 一、什么是压力测试二、压力测试关注点&#xff08;一&#xff09;压力测试分为两种测试场景&#xff08;二&#xff09;压测设置参数1.线程数&#xff1a;用于设置并发数量&#xff0c;也就是多少个用户同时访问2.Rame-Up Period(in seconds)&#xff1a;控制每隔多少…

广电手机卡靠谱吗?

广电手机卡&#xff0c;作为中国广播电视网络集团有限公司&#xff08;简称“中国广电”&#xff09;官方发行的手机卡&#xff0c;是中国第四大运营商推出的移动通信产品。其靠谱性可以从以下几个方面进行评估&#xff1a; 一、网络覆盖与信号质量 网络覆盖广泛&#xff1a;广…

MDK 复制hex文件到根目录并加上日期 bat 脚本

工程目录示例 copy.bat ::关闭命令行显示 echo off :: GBK chcp 936 >nul setlocal EnableDelayedExpansion:设置文件夹路径 set "FolderName.\Objects":: set "FolderName.\Objects" :: #;copy %FolderName%\*.hex ..\*.hex //复制到根目录 :: c…

C++ 洛谷 哈希表(对应题库:哈希,hash)习题集及代码

马上就开学了&#xff0c;又一个卷季&#xff0c;不写点东西怎么行呢&#xff1f;辣么&#xff0c;我不准备写那些dalao们都懂得&#xff0c;熟练的&#xff0c;想来想去&#xff0c;最终还是写哈希表吧&#xff01;提供讲解&题目&代码解析哦&#xff01; 奉上题目链接…

LVS的加权轮询算法

http://kb.linuxvirtualserver.org/wiki/Weighted_Round-Robin_Scheduling 加权轮循调度是为了更好地处理不同处理能力的服务器。每个服务器都可以被分配一个权重&#xff0c;一个表示处理能力的整数值。权值较高的服务器比权值较低的服务器首先接收到新连接&#xff0c;权值较…

Django异步查询并下载CSV文件

Django异步查询并下载CSV文件 通过循环遍历数据库,自动生成CSV文件的表头和内容. Django V5.1 1. 视图 1.1 将同步的数据库查询转换为异步 async def get_blogs():# 使用sync_to_async包装Queryset方法blog_list await sync_to_async(Blog.objects.all)()return blog_list…

心觉:你的潜意识信念系统符合第一性原理吗

想要释放潜意识的力量&#xff0c;以及想要吸引力法则发挥作用 每天进行积极的自我暗示非常重要 自我暗示辅以视觉化目标及实现目标后的喜悦&#xff0c;其实就是重塑潜意识的过程 举个例子&#xff0c;比如你现在月收入5000&#xff0c;你想实现月入5万 怎么做到 你现在月…

5.11 飞行控制——定点飞行

文章目录 5.11 飞行控制——定点飞行5.11.1 加入三轴位置的飞行硬件系统 FLY(s)5.11.2 数学模型——三轴位置系统&#xff08;1&#xff09;x、y轴位置系统的微分方程&#xff08;2&#xff09;z轴位置系统的微分方程&#xff08;3&#xff09;三轴位置系统的状态空间方程 5.11…

MybatisPlus的基本使用

文章目录 介绍特性工作流程图添加依赖Spring Boot2Spring Boot3 配置定义Mapper接口并继承BaseMapperServer 接口自定义 Service 接口继承 IServie 接口自定义 Service 实现类&#xff0c;实现自定义接口并继承 ServiceImpl 添加Config类常用注解:TableNameTableIdTableFieldTa…