我有以下 Groovy 脚本:
def n = ["1","2","3"]
println n.collect { v -> v.toInteger()*2 }
我想将其翻译为等效的 Java 代码(从“收集”的含义中抽象出来——现在它做什么并不重要)。
我写了以下内容:
class X {
X() {
object[] n = new object[]{"1","2","3"};
object anonymousBlock(object v) { return v.toInteger()*2; }
System.out.println(n.collect(???));
}
}
我应该使用上面的表示将什么作为参数传递给收集?
使用 Java 8,您可以:
List<String> strings = Arrays.asList( "1","2","3" ) ;
List<Integer> numbers = strings.stream()
.map( Integer::parseInt )
.map( (i) -> i * 2 )
.collect( Collectors.toList() ) ;
对于 Java 7,一种方法是这样的:
首先,定义一个接口以从一种类型转换为另一种类型:
static interface Mapper<T,U> {
U apply( T value ) ;
}
然后,我们可以定义一个委托给迭代器的类,并在返回的每个元素上应用
Mapper.apply
方法:
static class Collector<T,U> implements Iterator<U> {
Iterator<T> delegate ;
Mapper<T,U> mapper ;
public Collector( Iterable<T> elements, Mapper<T,U> c ) {
delegate = elements.iterator() ;
this.mapper = c ;
}
@Override
public void remove() { delegate.remove() ; }
@Override
public boolean hasNext() { return delegate.hasNext() ; }
@Override
public U next() { return mapper.apply( delegate.next() ) ; }
}
然后,我们可以使用如下方式调用它:
List<String> strings = Arrays.asList( "1","2","3" ) ;
// Create our mapping Iterator
Iterator<Integer> iter = new Collector<>( strings, new Mapper<String,Integer>() {
@Override
public Integer apply( String v ) {
return Integer.parseInt( v ) * 2 ;
}
} ) ;
// Collect back from iterator into a List
List<Integer> numbers = new ArrayList<>() ;
while( iter.hasNext() ) {
numbers.add( iter.next() ) ;
}
Java 8 和 Groovy FTW ;-)
请注意,对于现代的
List.of()
(自 Java 9 起)和 Stream.toList()
(自 Java 16 起),现在的答案看起来像
List<String> strings = List.of("1", "2", "3");
List<Integer> numbers = strings.stream()
.map(Integer::parseInt)
.map(i -> i * 2)
.toList();