corda/test/extra/Coroutines.java
2009-05-16 21:41:27 -06:00

78 lines
2.1 KiB
Java

package extra;
import static avian.Continuations.callWithCurrentContinuation;
import avian.CallbackReceiver;
import avian.Callback;
public class Coroutines {
private static void expect(boolean v) {
if (! v) throw new RuntimeException();
}
private static void produce(Consumer<Character> consumer) throws Exception {
consumer.consume('a');
consumer.consume('b');
consumer.consume('c');
}
private static void consume(Producer<Character> producer) throws Exception {
expect(producer.produce() == 'a');
expect(producer.produce() == 'b');
expect(producer.produce() == 'c');
}
public static void main(String[] args) throws Exception {
final CoroutineState<Character> state = new CoroutineState<Character>();
final Consumer<Character> consumer = new Consumer<Character>() {
public void consume(final Character c) throws Exception {
callWithCurrentContinuation(new CallbackReceiver() {
public Object receive(Callback continuation) {
state.produceNext = continuation;
state.consumeNext.handleResult(c);
throw new AssertionError();
}
});
}
};
final Producer<Character> producer = new Producer<Character>() {
public Character produce() throws Exception {
return callWithCurrentContinuation(new CallbackReceiver<Character>() {
public Character receive(Callback<Character> continuation)
throws Exception
{
state.consumeNext = continuation;
if (state.produceNext == null) {
Coroutines.produce(consumer);
} else {
state.produceNext.handleResult(null);
}
throw new AssertionError();
}
});
}
};
consume(producer);
}
private static class CoroutineState<T> {
public Callback produceNext;
public Callback<T> consumeNext;
}
private interface Producer<T> {
public T produce() throws Exception;
}
private interface Consumer<T> {
public void consume(T value) throws Exception;
}
}