该楼层疑似违规已被系统折叠 隐藏此楼查看此楼
在程序开发中,为对象的类型做强制转换是一种非常不友好的设计。但是在Java中,有些情况下,我们除了强制转换别无选择。幸好随着Java的发展,Java8对这类强制型的转换有了一定的改善。
我们先来看看Java8出现前,对象都是怎么被强制转换的:
类型强制转换
1、静态转型
Object obj; // may be an integer
if (obj instanceof Integer) {
Integer objAsInt = (Integer) obj;
// do something with 'objAsInt'
}
这段代码里使用了 instanceof 和转型操作符,这些操作符已经融入到语言当中了。对象转换的类型(这个例子中是Integer)必须是在编译期静态确定的,所以我们将这种转型称为静态转型。
如果obj不是Integer,上面的测试就会失败。如果我们以任何方式做类型转换,就会得到一个 ClassCastException 异常。如果obj是null,intanceof 测试会失败,但是转型是可以通过的,因为null可以被任何类型引用。
2、动态转型
有静态的转型,肯定就有动态的转型。有一种不常见的技术,即使用Class的方法,这些方法与上面的操作符的作用是一致的:
Object obj; // may be an integer
Class type = // may be Integer.class
if (type.isInstance(obj)) {
T objAsType = type.cast(obj);
// do something with 'objAsType'
}
因为转换的类型在编译期是不知道,所以我们将这种转型称之为动态转型。
对错误类型和 null 转型的测试结果,与静态转型的结果是完全一致的。
Java 8中的类型转换
看完以往的强制类型转换,我们来看看,经过改进后的Java 8 中的类型都是怎么转换的。
1、Stream及Optional的转型
对 Optional 中的值或 Stream 中的元素转型需要两个步骤:首先,需要过滤掉错误的类型,然后将其转换为目标类型。
Optional中的转型
Optional> obj; // may contain an Integer
Optional objAsInt = obj
.filter(Integer.class::isInstance)
.map(Integer.class::cast);
虽然,通过两个步骤就可完成转型,但是看起来有一点笨拙和冗余。
2、未来的对象转型
Java 8现有对象转型还略显笨拙和冗余买,但我们还是希望 Class的强制转型方法能返回一个 Optional 或者 Stream。如果传递的对象的类型是正确的,则返回一个包含该对象的Optional或Stream。否则返回的Optional或Stream不包含任何元素。
这钟方法的实现比较琐碎:
Class上的新方法
public Optional castIntoOptional(Object obj) {
if (isInstance(obj))
returnOptional.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 integers
Stream streamOfInts = stream.
flatMap(Integer.class::castIntoStream);
错误的实例类型或者null引用,在实例测试的时候会失败,所以返回空的 Optional 或 Stream。这种方式永远不会抛出 ClassCastException 异常。