- 浏览: 1681549 次
- 性别:
- 来自: 杭州699号
文章分类
最新评论
-
莫莫摸:
为什么不用dubbo
RCP数据传输模型回顾 -
大胡子爸爸:
String, Class 都实现了Serializable接 ...
RPC框架几行代码就够了 -
lss598018587:
谢谢大神分享,比起新手看复杂的dubbo框架还不如看大神的这一 ...
RPC框架几行代码就够了 -
15606915740:
你好,请问一下。<dubbo:consumer filt ...
Dubbo文档 -
joqk12345:
...
一些设计上的基本常识
经过多个版本的调整, CommonTemplate(http://www.commontemplate.org)的核心包设计逐渐稳定.
但访问者的设计一直是块心病, 并且访问者是合成模式[GoF95]树结构中比较重要的扩展点.
CommonTemplate中的访问者最开始设计:
其中, Node是Template, Element, Expression等的抽象. 如下:
Node是引擎实现的, 而Visitor是留给扩展者实现的.
大体分为:
1. 模板元素树和表达式树全遍历
2. 模板元素树全遍历(不访问表达式树)
3. 查找某一模板元素
4. 查找某一表达式元素(基于模板遍历)
如:
NodeCountVisitor (统计模板节点的个数)
TemplateDumpVisitor (导出模板结构)
DirectiveFindVisitor (查找指令)
VariableRequirementVisitor (计算模板所需的变量)
等等.
调用方式如:
在查找指令时通常不需要遍历指令表达式, 而访问者原始接口无法控制是否访问指令表达式.
重构:
加入访问控制值
这样, 可以用 if (node instanceof Expression) return SKIP; 控制不访问表达式树.
也可以通过return STOP; 停止访问.
当然, Node中的int accept(Visitor)方法也要返回和传递控制值:
然而, Visitor接口中单一的visit()方法强迫扩展者使用if(node instanceof XXX)语句判断类型, 丧失多态性.
重构:
将visit拆分, 依赖树的具体结点.
这样, 子类只要覆写需要的类型函数.
但过多的状态位控制流转, 也是一件不愉快的事.
并且表达式树与模板元素树两种类型没有区分.
重构:
拆分表达式树与模板元素树访问者,
并通过子类覆写的方式决定是否需要级联访问表达式树,
通过抛出StopVisitException运行时异常停止访问.
节点的accept也作相应处理:
但访问者的设计一直是块心病, 并且访问者是合成模式[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); }
发表评论
-
以HTTL为例讲讲模块分包&领域模型&扩展框架
2011-10-09 20:08 16462注:该博客内容已加入 ... -
CommonTemplate增加HTML标签版语法外套
2008-09-09 10:33 2927CommonTemplate(http://www.commo ... -
CommonTemplate发布0.8.6版本
2008-08-26 20:49 1767CommonTemplate发布0.8.6版本 ... -
CommonTemplate发布0.8.5版本
2008-08-04 13:23 1848CommonTemplate发布0.8.5版本(2008-08 ... -
CommonTemplate加入代码生成器
2008-07-21 13:15 2200模板引擎经常被用于做代码生成, 为此, CommonTempl ... -
加入对YAML数据格式的支持
2008-07-01 12:41 3963CommonTemplate(http://www.commo ... -
嵌套注释语法思考
2008-06-29 14:40 3922主流的C/C++/Java/C#等语言,都将注释语法设计成不可 ... -
转:开源协议
2008-06-10 17:23 2166来源:网络 (1)Contrib ... -
CommonTemplate完成查看器Viewer.exe(及安装程序)
2008-06-04 15:12 1837完成查看器初始版本. 实现功能: 双击*.ctl文件, 自动读 ... -
CommonTemplate完成外部构建树或表达式接口
2008-05-31 11:01 1914CommonTemplate: http://www.comm ... -
CommonTemplate异常国际化完成
2008-05-26 11:48 1897周未把一个累活给干了, 就是异常信息的国际化. 总共有220多 ... -
CommonTemplate加入对无穷数的支持.
2008-05-23 11:07 2713用"*"号表示无穷数, 常在下标号中使用, ... -
CommonTemplate导出模板所需变量结构
2008-05-12 18:28 2233在velocity的邮件列表中收到下面的邮件: Simon G ... -
CommonTemplate完成$snatch指令
2008-05-06 09:20 1863CommonTemplate(http://www.commo ... -
关于CTE当前API无法支持从非引擎方式构建模板树
2008-04-28 17:20 1763因隐藏了模板树的实现, 现在CommonTemplate(ht ... -
CommonTemplate完成DEBUG单步调试
2008-04-21 09:56 2484CommonTemplate(http://www.commo ... -
CommonTemplate准备加入$breakpoint指令
2008-04-19 10:30 2156准备在CommonTemplate( http://www.c ... -
很高兴桂林兄加入CommonTemplate的开发
2008-04-05 20:49 2896桂林的blog: http://jasongreen.itey ... -
展开式序列实现
2008-03-31 22:47 2047现在CommonTemplate(http://www.com ... -
CommonTemplate 0.8.3 版本发布
2008-03-31 15:05 2169项目地址: http://www.commontemplat ...
相关推荐
commontemplate框架,jsp模板化的利器 项目首页http://www.commontemplate.org/zh/index.html
commontemplate-0.8.1.jar,是使用jwebap所需要的jar包。没有它,启动会有错误
NULL 博文链接:https://javatar.iteye.com/blog/255801
私信博主免费获取真题解析以及代码
Screenshot_2024-05-10-20-21-01-857_com.chaoxing.mobile.jpg
数字图像处理|Matlab-频域增强实验-彩色图像的频域滤波.zip
2024-2030中国定向转向膜市场现状研究分析与发展前景预测报告
开源工时填报管理系统安装包
电子元件 电子行业 行业分析 数据分析 数据报告 行业报告
Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。
18-17.网站域名DNS被劫持,网站服务器密码被改.mp4
QYResearch:2023年前五大2,3,3',4'-联苯四甲酸二酐(α-BPDA)企业占据全球91%的市场份额.docx
2024-2030中国仿生智能餐饮机器人市场现状研究分析与发展前景预测报告
82-82.渗透测试-CVE-2017-8464“震网三代 反弹shell演示课件.mp4
Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。
渗透测试渗透测试之SQL注入基于报错注入(下)
Android是一种基于Linux内核(不包含GNU组件)的自由及开放源代码的移动操作系统,主要应用于移动设备,如智能手机和平板电脑。该系统最初由安迪·鲁宾开发,后被Google公司收购并注资,随后与多家硬件制造商、软件开发商及电信营运商共同研发改良。 Android操作系统的特点包括: 开放源代码:Android系统采用开放源代码模式,允许开发者自由访问、修改和定制操作系统,这促进了技术的创新和发展,使得Android系统具有高度的灵活性和可定制性。 多任务处理:Android允许用户同时运行多个应用程序,并且可以轻松地在不同应用程序之间切换,提高了效率和便利性。 丰富的应用生态系统:Android系统拥有庞大的应用程序生态系统,用户可以从Google Play商店或其他第三方应用市场下载和安装各种各样的应用程序,满足各种需求。 可定制性:Android操作系统可以根据用户的个人喜好进行定制,用户可以更改主题、小部件和图标等,以使其界面更符合个人风格和偏好。 多种设备支持:Android操作系统可以运行在多种不同类型的设备上,包括手机、平板电脑、智能电视、汽车导航系统等。 此外,Android系统还有一些常见的问题,如应用崩溃、电池耗电过快、Wi-Fi连接问题、存储空间不足、更新问题等。针对这些问题,用户可以尝试一些基本的解决方法,如清除应用缓存和数据、降低屏幕亮度、关闭没有使用的连接和传感器、限制后台运行的应用、删除不需要的文件和应用等。 随着Android系统的不断发展,其功能和性能也在不断提升。例如,最新的Android版本引入了更多的安全性和隐私保护功能,以及更流畅的用户界面和更强大的性能。此外,Android系统也在不断探索新的应用场景,如智能家居、虚拟现实、人工智能等领域。 总之,Android系统是一种功能强大、灵活可定制、拥有丰富应用生态系统的移动操作系统,在全球范围内拥有广泛的用户基础。
Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。
46-46.渗透测试-Kali Linux安全渗透.mp4
电子元件 电子行业 行业分析 数据分析 数据报告 行业报告