Cierre el escáner sin cerrar System.in

Estoy tratando de volver a factorizar una parte grande y de uso frecuente de mi aplicación en métodos separados para que sea más fácil de mantener.

Algunos de estos métodos le piden al usuario que ingrese y valida la entrada, así que he usado un escáner y un System.in pero cuando cierro el escáner también cierro System.in

Entonces mi pregunta es, ¿puedo evitar que System.in se cierre protegiéndolo con CloseShieldInputStream o debería simplemente comenzar a pasar un Scanner a los métodos?

Simplemente puede ignorar el cierre implementando el decorador personalizado.

public class UnClosableDecorator extends InputStream { private final InputStream inputStream; public UnClosableDecorator(InputStream inputStream) { this.inputStream = inputStream; } @Override public int read() throws IOException { return inputStream.read(); } @Override public int read(byte[] b) throws IOException { return inputStream.read(b); } @Override public int read(byte[] b, int off, int len) throws IOException { return inputStream.read(b, off, len); } @Override public long skip(long n) throws IOException { return inputStream.skip(n); } @Override public int available() throws IOException { return inputStream.available(); } @Override public synchronized void mark(int readlimit) { inputStream.mark(readlimit); } @Override public synchronized void reset() throws IOException { inputStream.reset(); } @Override public boolean markSupported() { return inputStream.markSupported(); } @Override public void close() throws IOException { //do nothing } } 

Y úsala en main

 public static void main(String[] args) throws Exception { System.setIn(new UnClosableDecorator(System.in)); } 

Simplemente use un FilterInputStream personalizado en lugar de System.in:

 new FilterInputStream(System.in) { @Override public void close() throws IOException { //don't close System.in! } } 

puedes dejarlo sin cerrarlo, simplemente establece la variable de retención en nulo