我有一个梦想,不再需要操作员和垂头丧气的instanceof ,却没有访客模式的笨拙和冗长。 所以我想出了以下DSL语法:

Object msg = //...whenTypeOf(msg).is(Date.class).    then(date -> println(date.getTime())).is(String.class).  then(str -> println(str.length())).is(Number.class).  then(num -> println(num.intValue())).orElse(obj -> println("Unknown " + obj));

在Java 8中,没有向下转换,简洁的语法,强类型的……完全可以实现的。使用lambda和一些泛型,我创建了一个名为typeof的小型库 ,它比instanceofVisitor模式结合在一起是干净的,易于使用且更健壮的。 优势包括:

  • 没有明确的垂头丧气
  • 避免instanceof
  • 清洁且易于使用
  • 强类型
  • 适用于我们无法控制的类,包括JDK

这个小实用程序是出于Akka和Java API的目的而开发的,目的是限制instanceof运算符的使用,但它更通用。 同样,您可以根据运行时类型返回一些信息:

int result = whenTypeOf(obj).is(String.class).thenReturn(String::length).is(Date.class).thenReturn(d -> (int) d.getTime()).is(Number.class).thenReturn(Number::intValue).is(TimeZone.class).thenReturn(tz -> tz.getRawOffset() / 1000).is(MyType.class).thenReturn(7).get();

该库从上到下检查每个is()子句,如果找到包括父类在内的第一个匹配类,则停止运行,因此is(Number.class)将同时匹配IntegerFloat 。 如果没有条件匹配,则调用get将失败,并带有异常。 您可以使用orElse()重写此行为orElse()比等效的is(Object.class)更容易阅读):

int result = whenTypeOf(obj).is(String.class).thenReturn(String::length).//...orElse(42);

DSL利用Java中的静态类型的优势,几乎不可能错误地使用该库-大多数错误会在编译期间立即捕获。 以下所有代码段甚至都不会编译:

//ERROR - two subsequent is()
whenTypeOf(obj).is(Foo.class).is(Bar.class)//ERROR - then() without prior is()
whenTypeOf(obj).then(x -> println(x))//ERROR - mixing then() and thenReturn()
whenTypeOf(obj).is(Foo.class).then(foo -> println(foo)).is(Bar.class).thenReturn(bar -> bar.getB());

基本上,您首先输入whenTypeOf()Ctrl + space会准确告诉您允许的内容。 使用静态类型语言设计类型安全且健壮的DSL的关键是尽可能地限制API,以便在编译时避免无效的状态和调用。 您最终将获得大量的小类 ,但这没关系,您的用户将不会看到此内容。 例如,签出FirstIs.java –第一次调用is()之后返回的对象:

public class FirstIs<S, T> {final Then<S> parent;private final S object;private final Class<T> expectedType;public Then<S> then(Consumer<T> thenBlock) {if (matchingType()) {thenBlock.accept(castObject());return new TerminalThen<>();}return parent;}public <R> ThenReturn<S, R> thenReturn(Function<T, R> result) {if (matchingType()) {return new TerminalThenReturn<>(object, result.apply(castObject()));}return new ThenReturn<>(object);}public <R> ThenReturn<S, R> thenReturn(R result) {if (matchingType()) {return new TerminalThenReturn<>(object, result);}return new ThenReturn<>(object);}//...}

编写DSL比使用DSL困难得多,但最终还是很有收获的。 注意如何使用不同的返回类型( Then vs. ThenReturn )只是为了确保在每个阶段只能访问有效的方法。 另一种方法是实现运行时检查(例如,您不编写is(...).is(...).then(...) )–但是为什么还要麻烦编译器为我们做呢?

希望您喜欢本文,如果您愿意在项目中尝试使用此实用程序,请告诉我。 它可以在GitHub上获得 。

参考:来自Java和社区博客的JCG合作伙伴 Tomasz Nurkiewicz提供的Java 8中的instanceof运算符和访客模式替换 。

翻译自: https://www.javacodegeeks.com/2013/10/instanceof-operator-and-visitor-pattern-replacement-in-java-8.html

Java 8中的instanceof运算符和访客模式替换相关推荐

  1. oracle无效的关系运算符_每日一课 | Java 8中的instanceof运算符和访客模式替换

    我有一个梦想,不再需要操作员和垂头丧气的instanceof ,却没有访客模式的笨拙和冗长.所以我想出了以下DSL语法: Object msg = //... whenTypeOf(msg).     ...

  2. java 访客模式,每日一课 | Java 8中的instanceof运算符和访客模式替换

    每日一课 | Java 8中的instanceof运算符和访客模式替换 每日一课 | Java 8中的instanceof运算符和访客模式替换 我有一个梦想,不再需要操作员和垂头丧气的instance ...

  3. instanceof运算符_Java 8中的instanceof运算符和访客模式替换

    instanceof运算符 我有一个梦想,不再需要操作员和垂头丧气的instanceof ,却没有访客模式的笨拙和冗长. 所以我想出了以下DSL语法: Object msg = //...whenTy ...

  4. Go设计模式--访客模式

    大家好,这里是每周都在陪你一起进步的网管-!今天继续学习设计模式-访客模式 访客模式也叫访问者模式(Visitor Pattern)是一种将数据结构对象与数据操作分离的设计模式,可以在不改变数据结构对 ...

  5. java instanceof运算符_Java 中的instanceof 运算符

    Java 中的instanceof 运算符是用来在运行时指出对象是否是特定类的一个实例.instanceof通过返回一个布尔值来指出,这个对象是否是这个特定类或者是它的子类的一个实例. 用法: res ...

  6. JavaScript中的instanceof运算符是什么?

    本文翻译自:What is the instanceof operator in JavaScript? The instanceof keyword in JavaScript can be qui ...

  7. java学习中,instanceof 关键字 和 final 关键字、值的传递(java 学习中的小记录)...

    java学习中,instanceof 关键字 和 final 关键字.值的传递(java 学习中的小记录)作者:王可利(Star·星星) instanceof 关键字 作用: 1.用来判断某个对象是否 ...

  8. java 访客模式,设计模式 - 访客模式( Visitor Pattern)

    设计模式 - 访客模式( Visitor Pattern) 在Visitor模式中,我们使用一个访问者类来更改元素类的执行算法. 通过这种方式,元素的执行算法可以随着访问者的变化而变化. 此模式属于行 ...

  9. unity3d 收费模式_unity3d中的访客模式

    unity3d 收费模式 抽象 (Abstract) Game development as a discipline is challenging on its own compared to tr ...

最新文章

  1. 特斯拉CEO对自动驾驶发表预测,专家:别扰乱公众的认知了
  2. Cassandra 单机入门例子——有索引
  3. 保存数组类型数据_JS基本数据类型和引用数据类型的区别及深浅拷贝
  4. 位运算和时间复杂度的分析
  5. 551. Student Attendance Record I 从字符串判断学生考勤
  6. twisted系列教程十六–twisted守护进程
  7. 一篇弄懂 HTTP和HTTPS基本关系
  8. mysql触发器_MySQL视图\触发器\事务初步认识
  9. iframe给php传值,向iframe传递参数
  10. MOQL—转换器(Translator)
  11. linux暗黑2黑屏,如何在Ubuntu 20.04中启用全局暗黑模式
  12. 什么才是真正的项目团队,我来告诉你需要做哪些
  13. fdfs-文件上传信息返回详情
  14. 卸载TeXLive2016
  15. Vue基础1-如何创建一个vue实例
  16. 【英语:基础进阶_原著扩展阅读】J1.英文原著的选择和有效阅读方法
  17. win10无法防问其他计算机没有权限,win10系统访问磁盘共享没有权限的解决方案...
  18. 训练GAN,你应该知道的二三事
  19. 当下推荐系统的分析和关于长尾效应的解决猜想
  20. E - Help Hanzo(LightOJ 1197)

热门文章

  1. php无get报错,php $_get报错怎么办
  2. mysql级联复制转换成一主两从_一主两从转级联复制
  3. 字节数组转jsonobject(如读取HttpServletRequest.inputstream到jsonobject)
  4. 官网下载mysql实例数据库
  5. 混合多云架构_使用混合多云每个人都应避免的3个陷阱(第3部分)
  6. junit5和junit4_JUnit 5 –条件
  7. 根据变量推断变量类型_Java A的新本地变量类型推断
  8. java代码自动生成的插件_如何使用插件生成自定义Java 8代码
  9. Spring Boot Actuator:在其顶部具有MVC层的自定义端点
  10. JUnit 5和Selenium –改善项目配置