是否可以在Java 8中强制转换流?

问题

是否可以在Java 8中强制转换流?假设我有一个对象列表,我可以做这样的事情来过滤掉所有其他对象:

Stream.of(objects).filter(c -> c instanceof Client)

在此之后,如果我想对客户做一些事情,我需要投下每个客户:

Stream.of(objects).filter(c -> c instanceof Client)
    .map(c -> ((Client) c).getID()).forEach(System.out::println);

这看起来有点难看。是否可以将整个流转换为其他类型?喜欢castStream<Object>to aStream<Client>

请忽略这样的事实,即做这样的事情可能意味着糟糕的设计。我们在计算机科学课上做这样的事情,所以我正在研究java 8的新功能,如果可能的话,我很好奇。


#1 热门回答(175 赞)

我不认为有办法开箱即用。一个可能更清洁的解决方案是:

Stream.of(objects)
    .filter(c -> c instanceof Client)
    .map(c -> (Client) c)
    .map(Client::getID)
    .forEach(System.out::println);

或者,如评论中所建议的那样,你可以使用cast方法 - 前者可能更容易阅读:

Stream.of(objects)
    .filter(Client.class::isInstance)
    .map(Client.class::cast)
    .map(Client::getID)
    .forEach(System.out::println);

#2 热门回答(13 赞)

沿着ggovan's answer,我这样做如下:

/**
 * Provides various high-order functions.
 */
public final class F {
    /**
     * When the returned {@code Function} is passed as an argument to
     * {@link Stream#flatMap}, the result is a stream of instances of
     * {@code cls}.
     */
    public static <E> Function<Object, Stream<E>> instancesOf(Class<E> cls) {
        return o -> cls.isInstance(o)
                ? Stream.of(cls.cast(o))
                : Stream.empty();
    }
}

使用这个辅助函数:

Stream.of(objects).flatMap(F.instancesOf(Client.class))
        .map(Client::getId)
        .forEach(System.out::println);

#3 热门回答(9 赞)

晚会,但我认为这是一个有用的答案。

flatMap将是最短的方式。

Stream.of(objects).flatMap(o->(o instanceof Client)?Stream.of((Client)o):Stream.empty())

Ifo是aClient然后使用单个元素创建Stream,否则使用空流。然后将这些流扁平化为aStream<Client>