如何以访客模式访问_重新访问了访客模式

如何以访客模式访问

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

问题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吗? 与(SELECT?FROM DUAL),在适当的字段实现可用时。 这就是使复合模式如此强大,通用的API和许多封装行为的组件,从而将行为的一部分委派给子组件的原因。

第2步负责API的演变

到目前为止,我们已经看到的复合模式非常直观,但是功能非常强大。 但是迟早,我们将需要更多的参数,因为我们发现要将状态从父级QueryPart传递给子级QueryParts。 例如,我们希望能够内联某些子句的绑定值。 也许某些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/355229.shtml

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

相关文章

abap 创建出口历程_SAP abap 需找出口(BADI)的几种方法

现在给出一些比较简单的方法首先,大家要知道,一个程序的出口不会太多,需找出口,很多的时候都是在尝试第二,方法:首先会给出事务码,然后通过SE93找出对应的程序Y第三,se16里面&#x…

Objective-C基础语法高速入门

Objective-C是Mac软件开发领域最基本的开发语言,假如我们对C语言已经非常熟悉或者具有面向对象语言的基础。对于我们学习Objective-C将会非常实用。 方法调用(Calling Methods) 为了可以尽快上手。我们先来看一些简单的样例。Objective-C语法里面主要的方法调用是这…

华为堡垒机_运维堡垒机----Gateone

简介:运维堡垒机的理念起源于跳板机。2000年左右,高端行业用户为了对运维人员的远程登录进行集中管理,会在机房里部署跳板机。跳板机就是一台服务器,维护人员在维护过程中,首先要统一登录到这台服务器上,然…

拼图推迟将Java 9的发布日期推迟到2017年

由于Jigsaw项目的延迟,Java 9的发布日期被推迟到2017年 由于项目延迟的悠久历史,这可能不足为奇,但是看起来备受期待的拼图项目已被延迟。 再次。 好消息是,与上一次使用Java 8不同,它仍在Java 9的开发路线上。坏消息…

前端获取当前url路径

前端获取当前url路径//获取url路径 function getUrlPath(){//获取当前网址var currentPath window.document.location.href;//获取主机地址之后的目录var pathName window.document.location.pathname;var pos currentPath.indexOf(pathName);//获取主机地址var localhostPa…

jquery改变字符串中部分字符的颜色

//该方法改变字符串中中括号内&#xff08;包括中括号&#xff09;的字符串颜色为红色function changecolocer() {  var zf $(#YWFA).text();   if(zf.length>0){     $(#YWFA).html(zf.replace(/\【.*?\】/g,<label style"color:red;">$&<…

java自动推断类型_Java 7的类型推断

java自动推断类型每个优秀的程序员都喜欢编写简洁但有效且经过优化的代码。 类型推断是JDK 7中引入的一种方法&#xff0c;它肯定会为您带来更少键入的好处。 您以以下方式使用Java代码已有很长时间了。 但是&#xff0c;在初始化Collections的特定实现时&#xff0c;您是否曾经…

shell 删除七日内日志_shell日志删除(超容量自动)

背景&#xff1a;避免双十一磁盘被打爆&#xff0c;本想通过crontab执行&#xff0c;但是删除需要密码&#xff0c;所以用作当机器磁盘高于摸个阈值&#xff0c;进行无关性日志强删#!/bin/sh#use#sh clean.sh wmporder_prehost 38(说明&#xff1a;磁盘超过38%&#xff0c;则进…

form表单序列化转换为json对象

form表单序列化转换为json对象//form表单序列化转换为json对象 (function($){$.fn.serializeJsonfunction(){var serializeObj{};var arraythis.serializeArray();$(array).each(function(){if(serializeObj[this.name]){if($.isArray(serializeObj[this.name])){serializeObj[…

拓扑排序 确定比赛名次

确定比赛名次Problem Description有N个比赛队&#xff08;1<N<500&#xff09;&#xff0c;编号依次为1&#xff0c;2&#xff0c;3&#xff0c;。。。。&#xff0c;N进行比赛&#xff0c;比赛结束后&#xff0c;裁判委员会要将所有参赛队伍从前往后依次排名&#xff0c…

JavaFX技巧22:“自动调整大小(树)”表列

JavaFX “缺少功能调查”中提到的“缺少功能”的第一件事就是能够自动调整表/树表中的列大小。 没错&#xff0c;没有公共API是正确的&#xff0c;但是当您密切关注时&#xff0c;您会注意到JavaFX内部一定有执行此操作的代码&#xff0c;因为用户可以通过双击分隔线自动调整列…

aesmiyao php_PHP使用AES,ECB模式块和PKCS5Padding生成对称密钥

首先,要对你需要即兴创作的输入做PKCS#5填充&#xff1a;// source: http://php.net/manual/en/ref.mcrypt.php#69782function pkcs5_pad($text, $blocksize){$pad $blocksize - (strlen($text) % $blocksize);return $text . str_repeat(chr($pad), $pad);}然后选择你的算法并…

解决复合主键

解决复合主键使用IdClass(BzdmKey.class)注解 package entity.po;import javax.persistence.*;/*** Package main.java.pojo* Description BZDM对象的pojo类* Author zhaohuaqing*/ Entity Table(name "ts_bzdm") IdClass(BzdmKey.class) public class Bzdm {/*** …

【数据挖掘导论】——数据质量

数据质量数据挖掘使用的数据一般是为其它用途收集或者收集的时候还没有明白目的。因此数据经常不能在数据的源头控制质量。为了避免数据质量的问题&#xff0c;所以数据挖掘着眼于两个方面&#xff1a;数据质量问题的检測和纠正&#xff08;数据清理&#xff09;&#xff1b;使…

rocksdb原理_[转]Rocksdb Compaction原理

概述compaction主要包括两类&#xff1a;将内存中imutable 转储到磁盘上sst的过程称之为flush或者minor compaction&#xff1b;磁盘上的sst文件从低层向高层转储的过程称之为compaction或者是major compaction。对于myrocks来说&#xff0c;compaction过程都由后台线程触发&am…

P2216 [HAOI2007]理想的正方形(二维RMQ)

题目描述 有一个a*b的整数组成的矩阵&#xff0c;现请你从中找出一个n*n的正方形区域&#xff0c;使得该区域所有数中的最大值和最小值的差最小。 输入输出格式 输入格式&#xff1a; 第一行为3个整数&#xff0c;分别表示a,b,n的值 第二行至第a1行每行为b个非负整数&#xff0…

MD5加密

MD5加密package common.util;import java.math.BigInteger; import java.security.MessageDigest;/*** Package main.java.utils* Description 加密* Author zhaohuaqing*/ public class MD5 {public static final String KEY_MD5 "MD5";/*** param inputStr 输入的…

jrockit_JRockit JRCMD教程

jrockit本文将为您提供概述和教程&#xff0c;说明如何使用jrcmd工具对JRockit Java Heap问题进行初始分析和问题隔离。 将来的文章中将介绍使用JRockit任务控制和堆转储分析&#xff08;仅JRockit R28 版&#xff09;的更深入的分析和教程。 有关JRockit Java堆空间的快速概述…

jQuery 事件 - ready() 方法

jQuery 事件 - ready() 方法当 DOM&#xff08;文档对象模型&#xff09; 已经加载&#xff0c;并且页面&#xff08;包括图像&#xff09;已经完全呈现时&#xff0c;会发生 ready 事件。 1.语法1 $(document).ready(function)2.语法2 $().ready(function)3.语法3 $(funct…

axios vue 加载效果动画_vue中使用axios拦截器实现数据加载之前的loading动画显示 @劉䔳...

首先新建一个 loading.vue组件&#xff0c;写loading动画效果.loader {width: 100%;height: 100%;display: flex;align-items: center;justify-content: center}-webkit-keyframes loading{50% {transform: scale(.4);opacity: .3}100% {transform: scale(1);opacity: 1}}.load…