重新访问了访客模式

访客模式是面向对象设计中最被高估但又被低估的模式之一。 高估了它,因为它常常被选择得太快( 可能是由建筑宇航员选择的 ),然后以错误的方式添加时会膨胀本来非常简单的设计。 如果您不遵循教科书示例,那么它可能会非常强大,因此被低估了。 让我们详细看一下。

问题1:命名

它的最大缺陷(在我看来)是命名本身。 “访客”模式。 当我们用google搜索它时,我们很可能会在相关的Wikipedia文章中找到自己,显示类似这样的有趣图像:

维基百科访客模式示例

对。 对于我们98%的人在日常软件工程工作中对车轮,发动机和车身的思考而言,这很明显,因为我们知道,机修工向我们收取几千美元的汽车维修费用后,我们会首先访问车轮,然后是发动机,然后最终访问我们的钱包并接受我们的现金。 如果我们很不幸,他也会在我们工作时拜访我们的妻子,但她永远不会接受那个忠实的灵魂。

但是,解决工作中其他问题的2%的人呢? 就像我们为电子银行系统,证券交易所客户,Intranet门户等编写复杂的数据结构时一样。为什么不将访客模式应用于真正的分层数据结构? 喜欢文件夹和文件? (好的,毕竟不是那么复杂)

好的,所以我们将“访问”文件夹,每个文件夹将让其文件“接受”为“访客”,然后我们也让访问者“访问”这些文件。 什么?? 汽车让其零件接纳访客,然后让访客自我访问吗? 这些条款具有误导性。 它们是通用的,适合设计模式。 但是它们会杀死您的现实设计,因为没有人会考虑“接受”和“访问”,而实际上您是在读/写/删除/修改文件系统。

问题2:多态

当应用于错误的情况时,这比命名引起的头痛甚至更多。 访客为什么在地球上认识其他所有人? 为什么访问者需要针对层次结构中每个涉及元素的方法? 多态性和封装要求将实现隐藏在API的后面。 (我们数据结构的)API可能以某种方式实现了复合模式 ,即其部分继承自公共接口。 好吧,当然,车轮不是汽车,我妻子也不是机械师。 但是当我们采用文件夹/文件结构时,它们不是全部都是java.util.File对象吗?

了解问题

实际的问题不是访问代码的命名和可怕的API详细程度,而是对模式的误解。 这不是最适合访问带有许多不同类型对象的大型复杂数据结构的模式。 这种模式最适合于访问几种不同类型的简单数据结构,但要访问数百名访问者。 取文件和文件夹。 那是一个简单的数据结构。 您有两种类型。 一个可以包含另一个,两者共享一些属性。 各种访客可能是:

  • CalculateSizeVisitor
  • FindOldestFileVisitor
  • DeleteAllVisitor
  • FindFilesByContentVisitor
  • ScanForVirusesVisitor
  • …你给它起名字

我仍然不喜欢命名,但是这种模式在这种范例中可以完美地工作。

那么,访客模式何时“错误”?

我想以jOOQ QueryPart结构为例。 其中有很多,可以对各种SQL查询结构进行建模,从而使jOOQ可以构建和执行任意复杂度的SQL查询。 让我们举几个例子:

  • 健康)状况
    • 组合条件
  • 领域
    • 表格栏位
  • 字段清单

还有更多。 它们中的每一个都必须能够执行两个操作:渲染SQL和绑定变量。 那将使两个访问者每个人都知道……40-50种类型……? 也许在遥远的将来,jOOQ查询将能够呈现JPQL或其他某种查询类型。 那将使3位访客面对40-50种类型。 显然,在这里,经典的访客模式是一个错误的选择。 但是我仍然想“访问” QueryPart,将渲染和绑定委托给较低的抽象级别。

那么如何实现呢?

很简单:坚持使用复合模式! 它允许您向每个人都必须实现的数据结构添加一些API元素。

因此,凭直觉,第一步就是

interface QueryPart {// Let the QueryPart return its SQLString getSQL();// Let the QueryPart bind variables to a prepared// statement, given the next bind index, returning// the last bind indexint bind(PreparedStatement statement, int nextIndex);
}

使用此API,我们可以轻松地抽象SQL查询并将职责委派给较低级别​​的工件。 例如,一个BetweenCondition。 它负责在[lower]和[upper]条件之间正确排序[field]的各部分,语法正确地呈现SQL,并将部分任务委派给其child-QueryParts:

class BetweenCondition {Field field;Field lower;Field upper;public String getSQL() {return field.getSQL() + ' between ' +lower.getSQL() + ' and ' +upper.getSQL();}public int bind(PreparedStatement statement, int nextIndex) {int result = nextIndex;result = field.bind(statement, result);result = lower.bind(statement, result);result = upper.bind(statement, result);return result;}
}

另一方面,BindValue主要负责变量绑定

class BindValue {Object value;public String getSQL() {return '?';}public int bind(PreparedStatement statement, int nextIndex) {statement.setObject(nextIndex, value);return nextIndex + 1;}
}

结合起来,我们现在可以轻松创建这种形式的条件: 在之间? 和?。 当实现更多QueryPart时,我们还可以想象像MY_TABLE.MY_FIELD BETWEEN吗? 如果可以使用适当的字段实现,则使用AND(选择?从双精度)。 这就是使复合模式如此强大,通用的API和许多封装行为的组件,从而将行为的一部分委派给子组件的原因。

第2步负责API的演变

到目前为止,我们已经看到了复合模式,非常直观,但是功能非常强大。 但是迟早,我们将需要更多的参数,因为我们发现要将状态从父级QueryPart传递给子级。 例如,我们希望能够内联某些子句的某些绑定值。 也许某些SQL方言不允许BETWEEN子句中的绑定值。 如何使用当前的API处理该问题? 扩展它,添加一个“布尔内联”参数? 没有! 这就是发明访客模式的原因之一。 为了使复合结构元素的API保持简单(只需执行“接受”)。 但是在这种情况下,用“上下文”替换参数比实现真正的访客模式好得多:

interface QueryPart {// The QueryPart now renders its SQL to the contextvoid toSQL(RenderContext context);// The QueryPart now binds its variables to the contextvoid bind(BindContext context);
}

上面的上下文包含这样的属性(setter和render方法返回上下文本身,以允许方法链接):

interface RenderContext {// Whether we're inlining bind variablesboolean inline();RenderContext inline(boolean inline);// Whether fields should be rendered as a field declaration// (as opposed to a field reference). This is used for aliased fieldsboolean declareFields();RenderContext declareFields(boolean declare);// Whether tables should be rendered as a table declaration// (as opposed to a table reference). This is used for aliased tablesboolean declareTables();RenderContext declareTables(boolean declare);// Whether we should cast bind variablesboolean cast();// Render methodsRenderContext sql(String sql);RenderContext sql(char sql);RenderContext keyword(String keyword);RenderContext literal(String literal);// The context's 'visit' methodRenderContext sql(QueryPart sql);
}

BindContext也是如此。 如您所见,该API相当可扩展,可以添加新属性,还可以添加其他常见的呈现SQL的方法。 但是BetweenCondition不必放弃有关如何呈现其SQL以及是否允许绑定变量的封装知识。 它会将这些知识保留给自己:

class BetweenCondition {Field field;Field lower;Field upper;// The QueryPart now renders its SQL to the contextpublic void toSQL(RenderContext context) {context.sql(field).keyword(' between ').sql(lower).keyword(' and ').sql(upper);}// The QueryPart now binds its variables to the contextpublic void bind(BindContext context) {context.bind(field).bind(lower).bind(upper);}
}

另一方面,BindValue主要负责变量绑定

class BindValue {Object value;public void toSQL(RenderContext context) {context.sql('?');}public void bind(BindContext context) {context.statement().setObject(context.nextIndex(), value);}
}

结论:将其命名为上下文模式,而不是访客模式

快速跳到访客模式时要小心。 在许多情况下,您将使设计变得肿,从而使其完全不可读且难以调试。 这里是要记住的规则,总结如下:

  1. 如果您有许多访问者并且数据结构相对简单(几种类型),那么访问者模式可能就可以了。
  2. 如果您有很多类型,并且访问者组相对较少(很少有行为),则访问者模式是过大的,请坚持使用复合模式
  3. 为了简化API的演变,请将您的复合对象设计为具有采用单个上下文参数的方法。
  4. 突然之间,您将再次遇到“几乎访问者”模式,其中context = visitor,“ visit”和“ accept” =“您专有的方法名称”

同时,“上下文模式”与“复合模式”一样直观,而与“访问者模式”一样强大,结合了两个方面的优势。

参考: 访问者模式是我们的JCG合作伙伴 Lukas Eder在JAVA,SQL和JOOQ博客上再次访问的 。


翻译自: https://www.javacodegeeks.com/2012/05/visitor-pattern-re-visited.html

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

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

相关文章

java web开发技术大_2021年六大javaweb开发主流技术

作为历史最为悠久的编程语言——java,历经数十年依然盘踞在编程榜最前面的位置,这与它的技术和应用范围是分不开的,同时呢,javaweb开发主流技术更是java开发者时时刻刻关注的问题,接下来我们一起分析一下2020年互联网行…

ASP.NET—013:实现带控件的弹出层(弹出框)

http://blog.csdn.net/yysyangyangyangshan/article/details/38458169 在页面中用到弹出新页面的情况比较多的,一般来说都是使用JS方法showModalDialog("新页面相对路径?参数1&参数2",window,"新页面样式");然后会新弹出一个模态的page页。…

运维人员日常工作(转自老男孩)

1)运维人员要谨记的6个字: 运维人员做事需遵循:简单、易用、高效 (2)运维人员服务的3大宗旨: 1、企业数据安全保障。 2、7*24小时业务持续提供服务。 3、不断提升用户感受、体验。 (3&#xff0…

c# 操作DatatTable

dtTemp.Columns.Add("列名");//增加一列 dtTemp.Columns.Remove("列名");//删除一列 dtTemp.Columns["旧列名"].ColumnName "新列名";//修改列名 dtTemp.Columns["列名1"].SetOrdinal(dtTemp.Columns["列名2"].O…

java 二进制 归属权限_【Java EE 学习 75 上】【数据采集系统第七天】【二进制运算实现权限管理】【权限分析和设计】...

一、权限计算相关分析1.如何存储权限首先说一下权限保存的问题,一个系统中最多有多少权限呢?一个大的系统中可能有成百上千个权限需要管理。怎么保存这么多的权限?首先,我们使用一个数字中的一位保存一种权限,那么如果…

MongoDB性能测试

因此,今天早上,我在mongo shell中四处乱逛。 我想出了三种不同的方式来聚合所需的数据,但不确定随后应移植哪种代码以在应用程序中使用。 那么,我将如何决定实施哪种方法呢? 好吧,让我们选择性能最佳的产品…

$_SERVER[SCRIPT_NAME]、$_SERVER[PHP_SELF]、$_SERVER[QUERY_STRING]、$_SERVER[REQUEST_URI]

1、$_SERVER["SCRIPT_NAME"] 说明:包含当前脚本的路径 2、$_SERVER["PHP_SELF"] 说明:当前正在执行脚本的文件名 3、$_SERVER["QUERY_STRING"] 说明:查询(query)的字符串 4、$_SERVER["REQUEST_URI"…

yii2增删改查及AR的理解

yii2增删改查 // 返回 id 为 1 的客户 $customer Customer::findOne(1); // 返回 id 为 1 且状态为 *active* 的客户 $customer Customer::findOne([ id > 1, status > Customer::STATUS_ACTIVE, ]); // 返回id为1、2、3的一组客户 $customers Customer::findAll([1, …

GWT和HTML5 Canvas演示

这是我对GWT和HTML5 Canvas的第一个实验。 我的第一个尝试是创建矩形,仅用几行代码就得出了这样的内容: 码: public class GwtHtml5 implements EntryPoint {static final String canvasHolderId "canvasholder";static final St…

mysql 平均值 排序_MySQL按平均两个平均值排序

我正在竞赛网站上工作,有两种类型的用户,普通网站成员和评委.每个人都可以使用拖放工具按照他们选择的顺序对特定比赛中的条目进行排序.完成后,相关的条目ID将附加一个排名值,然后可用于确定比赛中哪个条目获得最高的平均分数.获胜者实际上将通过平均每组的平均值来确定.我希望…

Solr管理界面详解

转载于:https://www.cnblogs.com/gslblog/p/6553813.html

iconv编码转换指令

看到一个不错的指令iconv,可以对文件编码进行转换,记录如下: iconv --list 列出所有支持转换的编码 icon -f code1 -t code2 filename -o newfile -f 即from 原来的编码 -t 即to 新的编码 filename 待转换的文件名 -o newfile 要输出的文件名 转载于:htt…

使用Spring Roo进行快速云开发–第2部分:VMware Cloud Foundry

Spring Roo是在Java平台上提供快速应用程序开发的工具。 我已经解释了何时使用它: http : //www.kai-waehner.de/blog/2011/04/05/when-to-use-spring-roo 。 Spring Roo目前支持两种针对云计算的解决方案:Google App Engine(GAE)…

java程序日期转换_Java 日期转换详解及实例代码

Java 日期转换涉及的核心类:Date类、SimpleDateFormat类、Calendar类一、 Date型与long型Date型转换为long型Date date new Date();//取得当前时间Date类型long date2long date.getTime();//Date转longlong型转换为Date型long cur System.currentTimeMills();//取…

软件设计之思想

编程用何种语言不重要,重要的是其设计思想。转载于:https://www.cnblogs.com/redfull/p/6554898.html

asp.net 与 java 2017_[ASP.net教程]C#与JAVA学习感悟

[ASP.net教程]C#与JAVA学习感悟0 2015-10-06 23:00:07C#与JAVA学习感悟学完C#与JAVA,感觉收获良多。C#与JAVA这两门语言相似度很高(了解它们早期历史的人可能知道为什么),也许很多人在学习JAVA(或C#)时会同时学习C#(或JAVA),因为它们太相似了…

Spring和JSF集成:国际化和本地化

如果您正在开发针对多种语言的JSF应用程序&#xff0c;那么您可能很熟悉<f&#xff1a;loadBundle>标记。 即使您的应用程序不支持使用消息包的国际化仍然是一个好主意。 在<f&#xff1a;loadBundle>标记下&#xff0c;它从Java java.util.ResourceBundle中读取消…

一个实用的却被忽略的命名空间:Microsoft.VisualBasic:

当你看到这个命名空间的时候&#xff0c;别因为是vb的东西就匆忙关掉网页&#xff0c;那将会是您的损失&#xff0c;此命名空间中的资源最初目的是为了简化vb.net开发而创建的&#xff0c;所以microsoft.visualbasic并不属于system命名空间&#xff0c;而是独立存在的。虽然是为…

Linux基础之命令练习Day2-useradd(mod,del),groupadd(mod,del),chmod,chown,

作业一&#xff1a; 1) 新建用户natasha&#xff0c;uid为1000&#xff0c;gid为555&#xff0c;备注信息为“master” 2) 修改natasha用户的家目录为/Natasha 3) 查看用户信息配置文件的最后一行 4) 为natasha用户设置密码“123” 5) 查看用户密码配置文件的最后一行 6) 将nat…

动态表单,JSF世界早已等待

新的PrimeFaces扩展版本0.5.0带来了新的DynaForm组件。 通常&#xff0c;如果知道行/列的数量&#xff0c;元素的位置等&#xff0c;则可以通过h&#xff1a;panelGrid或p&#xff1a;panelGrid来构建非常简单的表单。 对于静态表单&#xff0c;这是正确的。 但是&#xff0c;如…