详解设计模式:解释器模式

2022-12-09 15:07:05 浏览数 (1)

解释器模式(interpreter pattern),是在 GoF 23 种设计模式中定义了的行为型模式。 解释器模式 这种模式被用在 SQL 解析、符号处理引擎等。 解释器模式 给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。 解释器模式(Interpreter Pattern)提供了评估语言的语法或表达式的方式,它属于行为型模式。这种模式实现了一个表达式接口,该接口解释一个特定的上下文。 ~ 本篇文章内容包括:关于解释器模式、解释器模式 Demo(伪代码)


文章目录
  • 一、关于解释器模式
    • 1、关于解释器模式
    • 2、关于解释器模式的构成
    • 3、关于解释其模式的UML
    • 4、关于访问者模式的适用场景
    • 5、关于访问者模式的优缺点

  • 二、解释器模式 Demo(伪代码)
    • 1、伪代码 Demo 实现
    • 2、Demo 测试

一、关于解释器模式

1、关于解释器模式

解释器模式(interpreter pattern),是在 GoF 23 种设计模式中定义了的行为型模式。

解释器模式 这种模式被用在 SQL 解析、符号处理引擎等。

解释器模式 给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。 解释器模式(Interpreter Pattern)提供了评估语言的语法或表达式的方式,它属于行为型模式。这种模式实现了一个表达式接口,该接口解释一个特定的上下文。

2、关于解释器模式的构成

解释器模式包含以下主要 5 种角色。

  • 抽象表达式(Abstract Expression)角色:定义解释器的接口,约定解释器的解释操作,主要包含解释方法 interpret()。
  • 终结符表达式(Terminal Expression)角色:是抽象表达式的子类,用来实现文法中与终结符相关的操作,文法中的每一个终结符都有一个具体终结表达式与之相对应。
  • 非终结符表达式(Nonterminal Expression)角色:也是抽象表达式的子类,用来实现文法中与非终结符相关的操作,文法中的每条规则都对应于一个非终结符表达式。
  • 环境(Context)角色:通常包含各个解释器需要的数据或是公共的功能,一般用来传递被所有解释器共享的数据,后面的解释器可以从这里获取这些值。
3、关于解释其模式的UML
4、关于访问者模式的适用场景

解释器模式在实际的软件开发中使用比较少,因为它会引起效率、性能以及维护等问题。

在 Jdk 中的正则表达式中的 Pattern 类和 Spring 里面的 ExpressionParse 接口使用的是解释器模式的思想。

当一个语言需要解释执行,并且语言中的句子可以表示为一个抽象语法树的时候,如 XML 文档解释,整体来说还是一种应用较少的设计模式。

5、关于访问者模式的优缺点

# 访问者模式优点

  • 访问者模式增加新的操作很容易,只需要增加一个新的访问者即可;
  • 相关的行为,封装到一个访问者中;

# 访问者模式缺点

  • 增加新数据结构比较困难;
  • 元素变更比较困难,如为被访问的对象增加/减少一些属性,相应的访问者也需要进行修改。

二、解释器模式 Demo(伪代码)

1、伪代码 Demo 实现

# AbstractExpression 抽象表达式(Abstract Expression)角色

代码语言:javascript复制
public abstract class AbstractExpression {

    /**
     * 解释方法
     * @param info String
     * @return boolean
     */
    public abstract boolean interpret(String info);
}

# NonTerminalExpression 非终结符表达式(Nonterminal Expression)角色

代码语言:javascript复制
public class NonTerminalExpression extends AbstractExpression {

    private AbstractExpression address = null;
    private AbstractExpression name = null;
    private AbstractExpression id = null;

    public NonTerminalExpression(AbstractExpression address, AbstractExpression name, AbstractExpression id) {
        this.address = address;
        this.name = name;
        this.id = id;
    }

    @Override
    public boolean interpret(String info) {
        String[] strings = info.split("-");
        return address.interpret(strings[0]) && name.interpret(strings[1]) && id.interpret(strings[2]);
    }
}

# TerminalExpression 终结符表达式(Terminal Expression)角色

代码语言:javascript复制
import java.util.HashSet;
import java.util.Set;

public class TerminalExpression extends AbstractExpression {

    private final Set<String> set = new HashSet<String>();

    public TerminalExpression(String[] data) {
        for (int i = 0; i < data.length; i  )
            set.add(data[i]);
    }


    @Override
    public boolean interpret(String info) {
        if (set.contains(info)) {
            return true;
        }
        return false;
    }
}

# Context 环境(Context)角色

代码语言:javascript复制
public class Context {

    private final AbstractExpression information;

    public Context() {
        String[] number1 = {"1", "2", "3", "4", "5", "6", "7", "8", "9", "0"};
        AbstractExpression number = new TerminalExpression(number1);
        String[] xxZm1 = {"a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l"};
        AbstractExpression xxZm = new TerminalExpression(xxZm1);
        String[] dxZm1 = {"A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L"};
        AbstractExpression dxZm = new TerminalExpression(dxZm1);
        information = new NonTerminalExpression(number, xxZm, dxZm);
    }

    public void interpreter(String info) {
        boolean ok = information.interpret(info);
        if (ok) {
            System.out.println("正确! ["   info   "] 满足  [单个数字-单个小写-单个大写]  的条件");
        } else {
            System.out.println("错误! ["   info   "] 不满足  [单个数字-单个小写-单个大写]  的条件");
        }
    }

}
2、Demo 测试
代码语言:javascript复制
public class Client {
    
    public static void main(String[] args) {
        Context people = new Context();
        people.interpreter("2-a-A");
        people.interpreter("11-A-5");
        people.interpreter("你-好-吖");
        people.interpreter("2aA");
    }
}

0 人点赞