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 consumer) throws Exception { consumer.consume('a'); consumer.consume('b'); consumer.consume('c'); } private static void consume(Producer 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 state = new CoroutineState(); final Consumer consumer = new Consumer() { 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 RuntimeException("unreachable"); } }); } }; Producer producer = new Producer() { public Character produce() throws Exception { return callWithCurrentContinuation(new CallbackReceiver() { public Character receive(Callback continuation) throws Exception { state.consumeNext = continuation; if (state.produceNext == null) { Coroutines.produce(consumer); } else { state.produceNext.handleResult(null); } throw new RuntimeException("unreachable"); } }); } }; consume(producer); } private static class CoroutineState { public Callback produceNext; public Callback consumeNext; } private interface Producer { public T produce() throws Exception; } private interface Consumer { public void consume(T value) throws Exception; } }