第15章 解析器模式

本文内容来源于网络收集

作者:冰河
来源:冰河技术公众号

  • 本章难度:★★☆☆☆
  • 本章重点:用最简短的篇幅介绍解析器模式最核心的知识,理解解析器模式的设计精髓,并能够灵活运用到实际项目中,编写可维护的代码。

给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该文法表示来解释语言中的句子。

二、适用性

当有一个语言需要解释执行,并且可以将该语言中的句子表示为一个抽象语法树时,可使用解释器模式。而当存在以下情况时该模式效果最佳:

1.对于复杂的文法,文法的类层次变得庞大而无法管理。

2.关键的问题不是追求解释效率,最高效的解释器通常不是通过直接解释语法分析树实现的,而是首先将它们转换成另一种形式。

三、参与者

1.AbstractExpression(抽象表达式) 声明一个抽象的解释操作,这个接口为抽象语法树中所有的节点所共享。

2.TerminalExpression(终结符表达式) 实现与文法中的终结符相关联的解释操作。 一个句子中的每个终结符需要该类的一个实例。

3.NonterminalExpression(非终结符表达式) 为文法中的非终结符实现解释(Interpret)操作。

4.Context(上下文) 包含解释器之外的一些全局信息。

5.Client(客户) 构建(或被给定)表示该文法定义的语言中一个特定的句子的抽象语法树。 该抽象语法树由NonterminalExpression和TerminalExpression的实例装配而成。 调用解释操作。

四、类图


五、示例

AbstractExpression

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
/**
 * @author binghe(微信 : hacker_binghe)
 * @version 1.0.0
 * @description AbstractExpression
 * @github https://github.com/binghe001
 * @copyright 公众号: 冰河技术
 */
public abstract class Expression {
    public abstract void interpret(Context ctx);
}

Expression

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
/**
 * @author binghe(微信 : hacker_binghe)
 * @version 1.0.0
 * @description Expression
 * @github https://github.com/binghe001
 * @copyright 公众号: 冰河技术
 */
public class AdvanceExpression extends Expression{
    @Override
    public void interpret(Context ctx) {
        System.out.println("这是高级解析器!");
    }
}
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
/**
 * @author binghe(微信 : hacker_binghe)
 * @version 1.0.0
 * @description Expression
 * @github https://github.com/binghe001
 * @copyright 公众号: 冰河技术
 */
public class SimpleExpression extends Expression{
    @Override
    public void interpret(Context ctx) {
        System.out.println("这是普通解析器!");
    }
}

Context

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/**
 * @author binghe(微信 : hacker_binghe)
 * @version 1.0.0
 * @description Context
 * @github https://github.com/binghe001
 * @copyright 公众号: 冰河技术
 */
public class Context {

    private String content;
    private List<Expression> list = new ArrayList<>();
    public void setContent(String content) {
        this.content = content;
    }
    public String getContent() {
        return this.content;
    }
    public void add(Expression eps) {
        list.add(eps);
    }
    public List<Expression> getList() {
        return list;
    }
}

Test

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
/**
 * @author binghe(微信 : hacker_binghe)
 * @version 1.0.0
 * @description 测试类
 * @github https://github.com/binghe001
 * @copyright 公众号: 冰河技术
 */
public class Test {

    public static void main(String[] args) {
        Context ctx = new Context();
        ctx.add(new SimpleExpression());
        ctx.add(new AdvanceExpression());
        ctx.add(new SimpleExpression());
        for (Expression eps : ctx.getList()) {
            eps.interpret(ctx);
        }
    }
}

Result

1
2
3
这是普通解析器!
这是高级解析器!
这是普通解析器!