instanceof运算符_Java 8中的instanceof运算符和访客模式替换
instanceof運(yùn)算符
我有一個(gè)夢(mèng)想,不再需要操作員和垂頭喪氣的instanceof ,卻沒(méi)有訪客模式的笨拙和冗長(zhǎng)。 所以我想出了以下DSL語(yǔ)法:
在Java 8中,沒(méi)有向下轉(zhuǎn)換,簡(jiǎn)潔的語(yǔ)法,強(qiáng)類型的……完全可以實(shí)現(xiàn)的。使用lambda和一些泛型,我創(chuàng)建了一個(gè)名為typeof的小型庫(kù) ,它比instanceof和Visitor模式結(jié)合在一起是干凈的,易于使用且更健壯的。 優(yōu)勢(shì)包括:
- 沒(méi)有明確的垂頭喪氣
- 避免instanceof
- 清潔且易于使用
- 強(qiáng)類型
- 適用于我們無(wú)法控制的類,包括JDK
這個(gè)小實(shí)用程序是出于Akka和Java API的目的而開發(fā)的,目的是限制instanceof運(yùn)算符的使用,但它更通用。 同樣,您可以根據(jù)運(yùn)行時(shí)類型返回一些信息:
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();該庫(kù)從上到下檢查每個(gè)is()子句,如果找到包括父類在內(nèi)的第一個(gè)匹配類,則停止運(yùn)行,因此is(Number.class)將同時(shí)匹配Integer和Float 。 如果沒(méi)有條件匹配,則調(diào)用get將失敗,并帶有異常。 您可以使用orElse()重寫此行為orElse()比等效的is(Object.class)更容易閱讀):
int result = whenTypeOf(obj).is(String.class).thenReturn(String::length).//...orElse(42);DSL利用Java中的靜態(tài)類型的優(yōu)勢(shì),幾乎不可能錯(cuò)誤地使用該庫(kù)-大多數(shù)錯(cuò)誤會(huì)在編譯期間立即被發(fā)現(xiàn)。 以下所有代碼段甚至都不會(huì)編譯:
//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會(huì)告訴您所允許的內(nèi)容。 以靜態(tài)類型語(yǔ)言設(shè)計(jì)類型安全且健壯的DSL的關(guān)鍵是盡可能地限制API,以便在編譯時(shí)避免無(wú)效的狀態(tài)和調(diào)用。 您最終將獲得大量的小類 ,但這沒(méi)關(guān)系,您的用戶將看不到這一點(diǎn)。 例如,簽出FirstIs.java –第一次調(diào)用is()之后返回的對(duì)象:
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 )只是為了確保在每個(gè)階段只能訪問(wèn)有效的方法。 一種替代方法是執(zhí)行運(yùn)行時(shí)檢查(例如,您不編寫is(...).is(...).then(...) )–但是,如果編譯器可以為我們做這件事,為什么還要麻煩?
希望您喜歡本文,如果您愿意在項(xiàng)目中嘗試使用此實(shí)用程序,請(qǐng)告訴我。 它在GitHub上可用 。
翻譯自: https://www.javacodegeeks.com/2013/10/instanceof-operator-and-visitor-pattern-replacement-in-java-8.html
instanceof運(yùn)算符
總結(jié)
以上是生活随笔為你收集整理的instanceof运算符_Java 8中的instanceof运算符和访客模式替换的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 男友电脑里女尸的照片男友电脑里女尸的照片
- 下一篇: scala 异步调用_非阻塞异步Java