首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 软件管理 > 软件架构设计 >

CommonTemplate访问者设计考虑

2012-11-07 
CommonTemplate访问者设计思考经过多个版本的调整, CommonTemplate(http://www.commontemplate.org)的核心

CommonTemplate访问者设计思考
经过多个版本的调整, CommonTemplate(http://www.commontemplate.org)的核心包设计逐渐稳定.
但访问者的设计一直是块心病, 并且访问者是合成模式[GoF95]树结构中比较重要的扩展点.
CommonTemplate中的访问者最开始设计:

public interface Visitor {/** * 当访问到节点时被回调 * @param node 被访问的节点 */void visit(Node node);}

其中, Node是Template, Element, Expression等的抽象. 如下:
public interface Node {/** * 接收访问者, 并带领访问者遍历整个子树. (前序遍历) * @param visitor 访问者 */void accept(Visitor visitor);String getName();......}

Node是引擎实现的, 而Visitor是留给扩展者实现的.
大体分为:
1. 模板元素树和表达式树全遍历
2. 模板元素树全遍历(不访问表达式树)
3. 查找某一模板元素
4. 查找某一表达式元素(基于模板遍历)
如:
NodeCountVisitor (统计模板节点的个数)
TemplateDumpVisitor (导出模板结构)
DirectiveFindVisitor (查找指令)
VariableRequirementVisitor (计算模板所需的变量)
等等.
调用方式如:
Visitor visitor = new TemplateDumpVisitor(writer);template.accept(visitor); // 带领visitor遍历整个树, 遇到节点则回调visitor的相应方法

在查找指令时通常不需要遍历指令表达式, 而访问者原始接口无法控制是否访问指令表达式.
重构:
加入访问控制值
public interface Visitor {/** * 继续访问下一节点 */public static final int NEXT = 0;/** * 跳过子节点 */public static final int SKIP = 1;/** * 停止访问 */public static final int STOP = 2;/** * 当访问到节点时被回调 * @param node 被访问的节点 * @return 访问控制值, NEXT, SKIP, STOP */int visit(Node node);}

这样, 可以用 if (node instanceof Expression) return SKIP; 控制不访问表达式树.
也可以通过return STOP; 停止访问.
当然, Node中的int accept(Visitor)方法也要返回和传递控制值:
public interface Node {/** * 接收访问者, 并带领访问者遍历整个子树. (前序遍历) * @param visitor 访问者 * @return 访问控制值, Visitor.NEXT, Visitor.SKIP, Visitor.STOP */int accept(Visitor visitor);}

然而, Visitor接口中单一的visit()方法强迫扩展者使用if(node instanceof XXX)语句判断类型, 丧失多态性.
重构:
将visit拆分, 依赖树的具体结点.
public abstract class Visitor { // 考虑树的具体结点可能增加, 采用抽象类便于向前兼容public int visitDirective(Directive directive){}public int visitVariable(Variable variable){}......或者:public int visit(Directive directive){}public int visit(Variable variable){}......}

这样, 子类只要覆写需要的类型函数.
但过多的状态位控制流转, 也是一件不愉快的事.
并且表达式树与模板元素树两种类型没有区分.
重构:
拆分表达式树与模板元素树访问者,
并通过子类覆写的方式决定是否需要级联访问表达式树,
通过抛出StopVisitException运行时异常停止访问.
public abstract class ExpressionVisitor {/** * 当访问到变量时被回调 * * @param variable 访问到的变量 * @throws StopVisitException 当希望停止访问时抛出 */public void visitVariable(Variable variable) throws StopVisitException {}/** * 当访问到常量时被回调 * * @param constant 访问到的常量 * @throws StopVisitException 当希望停止访问时抛出 */public void visitConstant(Constant constant) throws StopVisitException {}/** * 当访问到二元操作符时被回调 * * @param binaryOperator 访问到的二元操作符 * @throws StopVisitException 当希望停止访问时抛出 */public void visitBinaryOperator(BinaryOperator binaryOperator) throws StopVisitException {}/** * 当访问到一元操作符时被回调 * * @param unaryOperator 访问到的一元操作符 * @throws StopVisitException 当希望停止访问时抛出 */public void visitUnaryOperator(UnaryOperator unaryOperator) throws StopVisitException {}}

public abstract class TemplateVisitor extends ExpressionVisitor {/** * 当访问到模板时被回调 * * @param template 访问到的模板 * @throws StopVisitException 当希望停止访问时抛出 */public void visitTemplate(Template template) throws StopVisitException {}/** * 模板访问结束时被回调 * * @param template 结束的模板 * @throws StopVisitException 当希望停止访问时抛出 */public void endTemplate(Template template) throws StopVisitException {}/** * 当访问到文本块或不解析块时被回调 * * @param text 访问到的文本块或不解析块 * @throws StopVisitException 当希望停止访问时抛出 */public void visitText(Text text) throws StopVisitException {}/** * 当访问到行注释或块注释时被回调 * * @param comment 访问到的行注释或块注释 * @throws StopVisitException 当希望停止访问时抛出 */public void visitComment(Comment comment) throws StopVisitException {}/** * 当访问到行指令时被回调.<br> * 注:缺省实现为继续访问指令表达式。<br> * 如果不需要访问指令表达式,请覆写此函数并留空。<br> * 也可以在访问指令表达式前后作相关处理:<br> * <pre> * public void visitDirective(Directive directive) { *     // 在表达式访问之前处理... *     super.visitDirective(directive); *     // 在表达式访问之后处理... * } * </pre> * @param directive 访问到的行指令 * @throws StopVisitException 当希望停止访问时抛出 */public void visitDirective(Directive directive) throws StopVisitException {if (directive.getExpression() != null)directive.getExpression().accept(this);}/** * 当访问到块指令时被回调.<br> * 注:缺省实现为继续访问指令表达式。<br> * 如果不需要访问指令表达式,请覆写此函数并留空。<br> * 也可以在访问指令表达式前后作相关处理:<br> * <pre> * public void visitBlockDirective(BlockDirective blockDirective) { *     // 在表达式访问之前处理... *     super.visitBlockDirective(blockDirective); *     // 在表达式访问之后处理... * } * </pre> * @param blockDirective 访问到的块指令 * @throws StopVisitException 当希望停止访问时抛出 */public void visitBlockDirective(BlockDirective blockDirective) throws StopVisitException {if (blockDirective.getExpression() != null)blockDirective.getExpression().accept(this);}/** * 块指令访问结束时被回调 * * @param blockDirective 结束的块指令 * @throws StopVisitException 当希望停止访问时抛出 */public void endBlockDirective(BlockDirective blockDirective) throws StopVisitException {}}

节点的accept也作相应处理:
public abstract class Node {/** * 接收访问者, 并带领访问者遍历整个子树. (前序遍历) * @param visitor 访问者 */void accept(Visitor visitor) {accept(visitor, true);}/** * 状态传入访问者接收接口, 通常直接使用accept(Visitor visitor) * @param visitor 访问者 * @param isEnter 是否为入口, 在入口处忽略StopVisitException */void accept(Visitor visitor, boolean isEnter);}
  

热点排行