对于简单的 Linq To Objects,我认为在 Java 中可以做到的最好的事情是这样的:
Vector<Integer> numbers = new Vector<Integer>();
numbers.add(42);
numbers.add(3);
numbers.add(16);
numbers.add(92);
numbers.add(9);
Iterable<Integer> filtered = new Where<Integer>(numbers) {
protected boolean predicate(Integer i) { return i > 10; }
};
Iterable<String> converted = new Select<Integer, String>(filtered) {
protected String select(Integer i) { return i.toString(); }
};
for (final String str : converted)
System.out.println(str);
请注意,我没有在一个表达式中将Where
和Select
链接在一起。我可以将 of 的定义插入filtered
它使用的一个地方,但这可能会使其(甚至)不那么可读。问题是缺少扩展方法和 lambdas。我们可以通过这些匿名类声明最接近 lambdas。它们可以引用在封闭范围内命名的对象,但只能引用final
s,因此它们不能改变任何东西(与 C# 中的 lambdas 不同)。
非常冗长的语法也是一种痛苦。人们经常建议 Java 应该为只有一个抽象(或接口)方法的情况提供更简单的语法,因此不需要为您想要覆盖的内容提供名称或类型声明。然后是没有类型推断的事实,也没有明显的方法在泛型类构造函数上提供它,因为new Select(filtered)
已经意味着其他东西。
Select
和的实现Where
是:
abstract class Select<TSource, TResult> implements Iterable<TResult>
{
private Iterable<TSource> _source;
public Select(Iterable<TSource> source)
{ _source = source; }
private class Iter implements Iterator<TResult>
{
private Iterator<TSource> _i;
public Iter() { _i = _source.iterator(); }
public void remove()
{ _i.remove(); }
public boolean hasNext()
{ return _i.hasNext(); }
public TResult next()
{ return select(_i.next()); }
}
protected abstract TResult select(TSource source);
public Iterator<TResult> iterator()
{ return new Iter(); }
}
abstract class Where<TSource> implements Iterable<TSource>
{
private Iterable<TSource> _source;
public Where(Iterable<TSource> source)
{ _source = source; }
private class Iter implements Iterator<TSource>
{
private Iterator<TSource> _i;
private TSource _cachedNext;
private boolean _hasCachedNext;
public Iter()
{
_i = _source.iterator();
fetch();
}
public void remove()
{ _i.remove(); }
public boolean hasNext()
{ return _hasCachedNext; }
public TSource next()
{
TSource result = _cachedNext;
fetch();
return result;
}
private void fetch()
{
_hasCachedNext = false;
while (_i.hasNext())
{
_cachedNext = _i.next();
if (predicate(_cachedNext))
{
_hasCachedNext = true;
return;
}
}
}
}
protected abstract boolean predicate(TSource source);
public Iterator<TSource> iterator()
{ return new Iter(); }
}