Thread interrupt not ending blocking call on input stream read

前端 未结 3 1328
时光取名叫无心
时光取名叫无心 2021-01-04 13:30

I\'m using RXTX to read data from a serial port. The reading is done within a thread spawned in the following manner:

CommPortIdentifier portIdentifier = Co         


        
相关标签:
3条回答
  • 2021-01-04 13:49

    i am using the code below to shutdown rxtx. i run tests that start them up and shut them down and the seems to work ok. my reader looks like:

    private void addPartsToQueue(final InputStream inputStream) {
        byte[] buffer = new byte[1024];
        int len = -1;
        boolean first = true;
        // the read can throw
        try {
            while ((len = inputStream.read(buffer)) > -1) {
                if (len > 0) {
                    if (first) {
                        first = false;
                        t0 = System.currentTimeMillis();
                    } else
                        t1 = System.currentTimeMillis();
                    final String part = new String(new String(buffer, 0, len));
                    queue.add(part);
                    //System.out.println(part + " " + (t1 - t0));
                }
                try {
                    Thread.sleep(sleep);
                } catch (InterruptedException e) {
                    //System.out.println(Thread.currentThread().getName() + " interrupted " + e);
                    break;
                }
            }
        } catch (IOException e) {
            System.err.println(Thread.currentThread().getName() + " " + e);
            //if(interruSystem.err.println(e);
            e.printStackTrace();
        }
        //System.out.println(Thread.currentThread().getName() + " is ending.");
    }
    

    thanks

    public void shutdown(final Device device) {
        shutdown(serialReaderThread);
        shutdown(messageAssemblerThread);
        serialPort.close();
        if (device != null)
            device.setSerialPort(null);
    }
    
    public static void shutdown(final Thread thread) {
        if (thread != null) {
            //System.out.println("before intterupt() on thread " + thread.getName() + ", it's state is " + thread.getState());
            thread.interrupt();
            //System.out.println("after intterupt() on thread " + thread.getName() + ", it's state is " + thread.getState());
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                System.out.println(Thread.currentThread().getName() + " was interrupted trying to sleep after interrupting" + thread.getName() + " " + e);
            }
            //System.out.println("before join() on thread " + thread.getName() + ", it's state is " + thread.getState());
            try {
                thread.join();
            } catch (InterruptedException e) {
                System.out.println(Thread.currentThread().getName() + " join interruped");
            }
            //System.out.println(Thread.currentThread().getName() + " after join() on thread " + thread.getName() + ", it's state is" + thread.getState());
        }
    
    0 讨论(0)
  • 2021-01-04 13:58

    You can't make a stream that doesn't support interruptible I/O into an InterruptibleChannel simply by wrapping it (and, anyway, ReadableByteChannel doesn't extend InterruptibleChannel).

    You have to look at the contract of the underlying InputStream. What does SerialPort.getInputStream() say about the interruptibility of its result? If it doesn't say anything, you should assume that it ignores interrupts.

    For any I/O that doesn't explicitly support interruptibility, the only option is generally closing the stream from another thread. This may immediately raise an IOException (though it might not be an AsynchronousCloseException) in the thread blocked on a call to the stream.

    However, even this is extremely dependent on the implementation of the InputStream—and the underlying OS can be a factor too.


    Note the source code comment on the ReadableByteChannelImpl class returned by newChannel():

      private static class ReadableByteChannelImpl
        extends AbstractInterruptibleChannel       // Not really interruptible
        implements ReadableByteChannel
      {
        InputStream in;
        ⋮
    
    0 讨论(0)
  • 2021-01-04 14:08

    The RXTX SerialInputStream (what is returned by the serial.getInputStream() call) supports a timeout scheme that ended up solving all my problems. Adding the following before creating the new SerialReader object causes the reads to no longer block indefinitely:

    serial.enableReceiveTimeout(1000);
    

    Within the SerialReader object, I had to change a few things around to read directly from the InputStream instead of creating the ReadableByteChannel, but now, I can stop and restart the reader without issue.

    0 讨论(0)
提交回复
热议问题