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

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

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

e61e284cf3ea50c8b922a5e02f381cc6.gif

我有一个梦想,不再需要操作员和垂头丧气的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的小型库 ,它比instanceof和Visitor模式结合在一起是干净的,易于使用且更健壮的。优势包括:

没有明确的垂头丧气

避免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)将同时匹配Integer和Float 。如果没有条件匹配,则调用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 {

final Then parent;

private final S object;

private final Class expectedType;

public Then then(Consumer thenBlock) {

if (matchingType()) {

thenBlock.accept(castObject());

return new TerminalThen<>();

}

return parent;

}

public  ThenReturn thenReturn(Function result) {

if (matchingType()) {

return new TerminalThenReturn<>(object, result.apply(castObject()));

}

return new ThenReturn<>(object);

}

public  ThenReturn 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

今日福利

3bb35cb546a07933722264f9da0b3092.png

点击阅读原文,一起玩耍

每日一课 | Java 8中的instanceof运算符和访客模式替换相关教程

JAVA方式对接FIL节点钱包

JAVA方式对接FIL节点钱包 官网Lotus JSON-RPC API 链接: https://docs.filecoin.io/reference/lotus-api/#endpoints 完整RPC源码文档 链接: https://pkg.go.dev/github.com/filecoin-project/lotus/api?tab=doc#FullNode 实现下面之前需要先搭建FIL钱包节点

详解构建可运行的JavaScript规范的方法

编程不仅仅是给计算机下达如何完成一项任务的指令,它还包括以一种精确的方式与他人交流思想,甚至是与未来的自己。这样的交流可以有多个目标,也许是为了共享信息,或者只是为了更容易地修改—如果你不理解或不记得很久以前做过什么,那么就很难修改。 当我

Java类的加载及单亲委派模型

Java类的加载及单亲委派模型 类的生命周期: 加载-链接(验证+准备+解析)-初始化(使用前的准备)-使用-卸载。 其中,类的加载过程是 加载-链接(验证+准备+解析)-初始化(使用前的准备) 验证是验证代码合法性 准备是创建静态变量以及设置静态变量的初始值

[适合初中级Java程序员修炼手册从0搭建整个Web项目]

[适合初中级Java程序员修炼手册从0搭建整个Web项目] 前言 文本已收录至我的GitHub仓库,欢迎Star:https://github.com/bin392328206 种一棵树最好的时间是十年前,其次是现在 six-finger-web 一个Web后端框架的轮子从处理Http请求【基于Netty的请求级Web服务

Github上42.4K的JAVA跑动起来、调试、教你看代码

Github上42.4K的JAVA跑动起来、调试、教你看代码 mall项目是一套电商系统,包括前台商城系统及后台管理系统,看完代码对自己能力将有非常大的提高,为了更好的让大家心里明白后端究竟该做什么,前端项目就直接废弃,我们全部改用在线调用接口的方式进行调试处

【Java 面试题系列 05】Class类的作用是什么?如何获取Class对象

【Java 面试题系列 05】Class类的作用是什么?如何获取Class对象? 点击上方“ 方才编程 ”,每天进步一点点! 决定每天上午8点整,分享一道 Java 面试题。厚积薄发,每天进步一点点! 【第 5 天】题目 Class类的作用是什么?如何获取Class对象? 参考答案 Cl

LeetCode每日一题!!(LCP 01.猜数字)

LeetCode每日一题!!(LCP 01.猜数字) 题目 :小A 和 小B 在玩猜数字。小B 每次从 1, 2, 3 中随机选择一个,小A 每次也从 1, 2, 3 中选择一个猜。他们一共进行三次这个游戏,请返回 小A 猜对了几次? 输入的guess数组为 小A 每次的猜测,answer数组为 小B

JavaTCP连接

JavaTCP连接 输入输出操作 可以这样理解: 服务器操作 import java.io.*;import java.net.ServerSocket;import java.net.Socket;public class Server { private static int port = 8002;//设置端口号 public static void main(String[] args)throws IOExcepti

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值