Java 8型別轉換及改進
為物件的型別做強制轉換是一種非常不好的設計。但在某些情況下,我們沒有其他選擇。Java自誕生的那一天起,就具備這種功能。
我認為Java 8在一定程度改善了這項古老的技術。
靜態轉型
Java中最常用的轉型方式如下:
靜態轉型
Object obj; // may be an integerif (obj instanceof Integer) { Integer objAsInt = (Integer) obj; // do something with 'objAsInt'}
這裡使用了 instanceof 和轉型運算子,這些運算子已經融入到語言當中了。物件轉換的型別(這個例子中是Integer)必須是在編譯期靜態確定的,所以我們將這種轉型稱為靜態轉型。
如果obj不是Integer,上面的測試就會失敗。如果我們以任何方式做型別轉換,就會得到一個 ClassCastException 異常。如果obj是null,intanceof 測試會失敗,但是轉型是可以透過的,因為null可以被任何型別引用。
動態轉型
有一種不常見的技術,即使用Class的方法,這些方法與上面的運算子的作用是一致的。
動態轉換成已知型別
Object obj; // may be an integerif (Integer.class.isInstance(obj)) { Integer objAsInt = Integer.class.cast(obj); // do something with 'objAsInt'
注意,這個例子中型別的轉換也是在編譯期確定的,所以沒有必要這麼去做。
動態轉型
Object obj; // may be an integerClasstype = // may be Integer.classif (type.isInstance(obj)) { T objAsType = type.cast(obj); // do something with 'objAsType'}
因為轉換的型別在編譯期是不知道,所以我們將這種轉型稱之為動態轉型。
對錯誤型別和 null 轉型的測試結果,與靜態轉型的結果是完全一致的。
Stream及Optional的轉型
現在
對 Optional 中的值或 Stream 中的元素轉型需要兩個步驟:第一步,我們需要過濾掉錯誤的型別,然後我們需要將其轉換為目標型別。
Optional中的轉型
Optional> obj; // may contain an IntegerOptionalobjAsInt = obj .filter(Integer.class::isInstance) .map(Integer.class::cast);
我們需要兩個步驟來完成轉型,這雖然不是什麼大問題,但是我感覺還是有一點笨拙和冗餘。
未來(可能)
我建議Class的強制轉型方法能返回一個 Optional 或者 Stream。如果傳遞的物件的型別是正確的,則返回一個包含該物件的Optional或Stream。否則返回的Optional或Stream不包含任何元素。
這些方法的實現比較瑣碎:
Class上的新方法
public OptionalcastIntoOptional(Object obj) { if (isInstance(obj)) return Optional.of((T) obj); else Optional.empty();}public Stream castIntoStream(Object obj) { if (isInstance(obj)) return Stream.of((T) obj); else Stream.empty();}
我們可以使用 flatMap 一步完成過濾和強制轉換:
FlatMap的實現:
Stream> stream; // may contain integersStreamstreamOfInts = stream. flatMap(Integer.class::castIntoStream);
錯誤的例項型別或者null引用,在例項測試的時候會失敗,所以返回空的 Optional 或 Stream。這種方式永遠不會丟擲 ClassCastException 異常。
成本和收益
我們怎麼來衡量這些方法是否真正有用呢?
有多少程式碼真正會使用它們?
對於一箇中等水平的開發者來說,它們是否能提高程式碼的可讀性?
是否值得為其節約一行程式碼?
實現和維護它們的成本是多少?
我對這些問題的回答是:不多,是非常少。所以,這是一個總和趨近於0的遊戲,但是,我可以證明雖然收益不多,但卻是大於0的。
你怎麼認為的呢?你自己會使用這些方法嗎?
原文連結: 翻譯: -譯文連結:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/200/viewspace-2805469/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Java資料型別及型別轉換Java資料型別
- java- 型別-轉換:基本型別以及包裝型別的轉換Java型別
- 資料型別及轉換資料型別
- 【Java】資料型別轉換Java資料型別
- java中的型別轉換Java型別
- Delegate如何進行型別轉換?型別
- 2、java資料型別轉換Java資料型別
- JAVA型別轉換怎麼變Java型別
- 型別轉換型別
- scala和java資料型別轉換Java資料型別
- Java資料型別自動轉換(++ ,+=)Java資料型別
- java 基本型別的轉換規則Java型別
- Java註釋,識別符號,資料型別,型別轉換Java符號資料型別
- 資料型別,型別轉換資料型別
- java 基礎型別與byte[]的轉換Java型別
- Java中String和byte型別互相轉換Java型別
- [Java基礎]之 資料型別轉換Java資料型別
- Java資料型別的顯式轉換和隱式轉換Java資料型別
- js型別轉換JS型別
- 型別轉換(cast)型別AST
- Convert型別轉換型別
- Java入門系列-05-資料型別和型別轉換Java資料型別
- [基礎] JavaScript 型別轉換及面試題JavaScript型別面試題
- 兄弟連go教程(4)型別-引用及轉換Go型別
- 自學java筆記I 基本型別+轉義字元+資料型別的轉換Java筆記字元資料型別
- java基本型別和物件之間的轉換Java型別物件
- java基本資料型別與自動轉換Java資料型別
- JNI常用型別轉換型別
- 容器,型別轉換。List。型別
- c++ 型別轉換C++型別
- interface{} 型別的轉換型別
- 型別轉換注意點型別
- 變數型別轉換變數型別
- Spring型別轉換(Converter)Spring型別
- 資料型別轉換資料型別
- golang的型別轉換Golang型別
- 型別轉換運算子型別
- C# 型別轉換C#型別