sclass AllOnAll<A, B> { new L<A> aList; new L<B> bList; LPair<IntRange> todo = new LinkedList; LPair<A, B> todo2 = new LinkedList; synchronized void newA(A a) { add(aList, a); addPair(todo, intRange_last(aList), intRange(0, l(bList))); } synchronized void newB(B b) { add(bList, b); addPair(todo, intRange(0, l(aList)), intRange_last(bList)); } synchronized Pair<A, B> next() { while (empty(todo2)) { if (empty(todo)) null; Pair<IntRange> p = popFirst(todo); addAll(todo2, outerProduct_pairs(subList(aList, p.a), subList(bList, p.b))); } ret popFirst(todo2); } }