Tuesday, February 8, 2011

Why does "piping" a CharBuffer hang?

Why does the following method hang?

public void pipe(Reader in, Writer out) {
    CharBuffer buf = CharBuffer.allocate(DEFAULT_BUFFER_SIZE);
    while( in.read(buf) >= 0 ) {
      out.append(buf.flip());
    }
}
  • Answering my own question: you have to call buf.clear() between reads. Presumably, read is hanging because the buffer is full. The correct code is

    public void pipe(Reader in, Writer out) {
        CharBuffer buf = CharBuffer.allocate(DEFAULT_BUFFER_SIZE);
        while( in.read(buf) >= 0 ) {
          out.append(buf.flip());
          buf.clear();
        }
    }
    
    Steve Jessop : Hanging? Looking at the Readable API, I'd expect it to busy loop returning 0 on each read if the buffer is full, but that's a guess since it doesn't explicitly say what 'attempts to read' means. Hanging might mean that because you're multiply-writing the output, you've blocked the output.
    Chris Conway : I didn't check the detailed behavior of the loop. Either read blocks or repeatedly returns 0. The net effect: the loop never terminates.
    Steve Jessop : I guess for blocking to happen the output stream must be something with a finite capacity that isn't being drained, like a circular buffer - pretty rare case. Maybe I'm idiosyncratic in that I generally use "hang" to mean a deadlock rather than a livelock.
  • I would assume that it is a deadlock. The in.read(buf) locks the CharBuffer and prevents the out.append(buf) call.

    That is assuming that CharBuffer uses locks (of some kind)in the implementation. What does the API say about the class CharBuffer?

    Edit: Sorry, some kind of short circuit in my brain... I confused it with something else.

0 comments:

Post a Comment