본문으로 건너뛰기

try-with-resources

커넥션, 입출력 스트림과 같은 자원을 사용한 후에는 자원을 해제해서 성능 문제, 메모리 누수 등을 방지해야 합니다. try-with-resources는 이러한 자원을 자동으로 해제하는 기능으로, java 7부터 도입되었습니다.

try-with-resources가 정상적으로 동작하려면 AutoCloseable 인터페이스를 구현한 객체를 사용해야 하고, try() 괄호 내에서 변수를 선언해야 합니다.

try (BufferedReader br = new BufferedReader(new FileReader("path"))) {
return br.readLine();
} catch (IOException e) {
return null;
}

java.io.BufferedReaderAutoCloseable 인터페이스를 구현하고 있습니다.

public class BufferedReader extends Reader {}

public abstract class Reader implements Readable, Closeable {}

public interface Closeable extends AutoCloseable {
public void close() throws IOException;
}

public interface AutoCloseable {
void close() throws Exception;
}

✔️ try-catch-finally 대신 try-with-resources를 사용해야 하는 이유는 뭘까?

BufferedReader br = null;
try {
br = new BufferedReader(new FileReader("path"));
return br.readLine();
} catch (IOException e) {
return null;
} finally {
if (br != null) {
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}

try-catch-finallyfinally 블록에서 close()를 명시적으로 호출해야 합니다. 하지만 close() 호출을 누락하거나 이 과정에서 또 다른 예외가 발생하면 예외 처리가 복잡해지는 문제가 있습니다.

또한 여러 개의 자원을 다룰 경우, 먼저 close()를 호출한 자원에서 에러가 발생하면 다음에 close()를 호출한 자원은 해제되지 않습니다. 이를 해결하려면 추가적인 try-catch-finally가 필요하기 때문에 가독성이 떨어지고, 실수할 가능성이 높습니다.

try-with-resourcestry-catch-finally의 문제를 해결 가능

  • try 블록이 종료될 때 close()를 자동으로 호출해서 자원을 해제합니다.
  • finally 블록 없이도 자원을 안전하게 정리하기 때문에 코드가 간결해집니다.
  • try 문에서 여러 자원을 선언하면, 선언된 반대 순서로 자동 해제됩니다.

✔️ Suppressed Exception(억제된 예외)란?

Suppressed Exception은 예외가 발생했지만 무시되는 예외를 의미합니다.

try-with-resourcesclose() 과정에서 발생한 예외를 Suppressed Exception으로 관리합니다.

class CustomResource implements AutoCloseable {

@Override
public void close() throws Exception {
throw new Exception("Close Exception 발생");
}

void process() throws Exception {
throw new Exception("Primary Exception 발생");
}
}

public class Main {

public static void main(String[] args) throws Exception {
try (CustomResource resource = new CustomResource()) {
resource.process();
}
}
}

Exception in thread "main" java.lang.Exception: Primary Exception 발생 at CustomResource.process(CustomResource.java:9) at Main.main(Main.java:5) Suppressed: java.lang.Exception: Close Exception 발생 at CustomResource.close(CustomResource.java:5) at Main.main(Main.java:4)

Suppressed Exception이 필요한 이유는 원래 예외(Primary Exception)를 유지하면서 추가 예외도 함께 추적할 수 있고, 자원을 안전하게 해제하면서 예외를 효율적으로 처리할 수 있습니다.

❗️try-catch-finallyclose()를 호출할 때 예외가 발생하면 원래 예외가 사라지고 close()에서 발생한 예외만 남을 수 있습니다.

public class Main {

public static void main(String[] args) throws Exception {
CustomResource resource = null;
try {
resource = new CustomResource();
resource.process();
} finally {
if (resource != null) {
resource.close();
}
}
}
}

Exception in thread "main" java.lang.Exception: Close Exception 발생 at CustomResource.close(CustomResource.java:5) at Main.main(Main.java:16)

이처럼 원래 예외가 사라지면 디버깅이 어려워질 수 있습니다. ThrowableaddSuppressed()를 사용하면 문제를 해결할 수 있지만 코드가 더욱 복잡해지기 때문에 try-with-resources를 사용하는 것이 좋습니다.

Loading comments...