微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何包装java.util.Iterator以更改正在迭代的对象的类型

我正在为第三方API提供一些外观类,我需要包装一个迭代器,以便我可以替换用我自己的外观对象迭代的东西.

这是我的facade类的简化版本,它包装了一个名为Item的API类

class FacadeItem {
    Item item;
    FacadeItem(Item item) {
        this.item = item;
    }
}

API提供此形式的迭代器Iterator< Item>

我需要实现这个形式的迭代器Iterator< FacadeItem>由API的迭代器支持.

我考虑使用Guava库中的ForwardingIterator如下:

class FacadeItemIterator<FacadeItem> extends ForwardingIterator<Item> {

    final Iterator<Item> delegate; // backing iterator

    FacadeItemIterator(Iterator<Item> delegate) {
        this.delegate = delegate;
    }

    @Override protected Iterator<Item> delegate() {
        return delegate;
    }

    @Override
    public FacadeItem next() {
        return new FacadeItem(super.next());
    }
}

但是编译器不允许覆盖next(),因为它期望返回的类型是Item,而不是FacadeItem

解决方法

Iterator接口不是那么大,所以你可以编写自己的委托迭代器:

class FacadeIterator implements Iterator<FacadeItem> {
    private final Iterator<Item> delegate; // set in ctor

    @Override
    public FacadeItem next() {
        return new FacadeItem(delegate.next());
    }

    // the other two methods just delegate straight
}

然后

Iterator<FacadeItem> facadeIterator = new FacadeIterator(itemIterator);

如果你正在使用Guava,你可以使用他们的Iterators.transform静态方法为你制作其中一个

Iterator<FacadeItem> facadeIterator = Iterators.transform(itemIterator,/* anon class of Function<Item,FacadeItem> */);

在Java 1.8中,这个选项变得非常简单:

Iterator<FacadeItem> facadeIterator
    = Iterators.transform(itemIterator,FacadeItem::new)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐