Java

try-with-resources는 왜 유용할까? (부제: 바이트 코드를 까보자)

Jaime.Lee 2023. 5. 16. 05:09

try-with-resources란?

자바에서 자원을 사용하는 블록을 처리할 때 자동으로 자원을 해제하는데 사용되는 구문으로 사용 방법은 아래와 같습니다.

try (Resource resource = new Resource()) {
    // 자원 사용 코드
} catch (Exception e) {
    // 예외 처리 코드
}

이 때 try의 매개변수에서 선언할 수 있으려면 AutoClosable 인터페이스를 구현해야 합니다. 대부분의 자바 표준 라이브러리 클래스들은 AutoClosable 인터페이스를 구현하고 있습니다. 커스텀 클래스를 사용하려면 해당 클래스가 AutoClosable 인터페이스를 구현해야 합니다.

AutoClosable 펼쳐 보기
/*
 * Copyright (c) 2009, 2013, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package java.lang;

/**
 * An object that may hold resources (such as file or socket handles)
 * until it is closed. The {@link #close()} method of an {@code AutoCloseable}
 * object is called automatically when exiting a {@code
 * try}-with-resources block for which the object has been declared in
 * the resource specification header. This construction ensures prompt
 * release, avoiding resource exhaustion exceptions and errors that
 * may otherwise occur.
 *
 * @apiNote
 * <p>It is possible, and in fact common, for a base class to
 * implement AutoCloseable even though not all of its subclasses or
 * instances will hold releasable resources.  For code that must operate
 * in complete generality, or when it is known that the {@code AutoCloseable}
 * instance requires resource release, it is recommended to use {@code
 * try}-with-resources constructions. However, when using facilities such as
 * {@link java.util.stream.Stream} that support both I/O-based and
 * non-I/O-based forms, {@code try}-with-resources blocks are in
 * general unnecessary when using non-I/O-based forms.
 *
 * @author Josh Bloch
 * @since 1.7
 */
public interface AutoCloseable {
    /**
     * Closes this resource, relinquishing any underlying resources.
     * This method is invoked automatically on objects managed by the
     * {@code try}-with-resources statement.
     *
     * <p>While this interface method is declared to throw {@code
     * Exception}, implementers are <em>strongly</em> encouraged to
     * declare concrete implementations of the {@code close} method to
     * throw more specific exceptions, or to throw no exception at all
     * if the close operation cannot fail.
     *
     * <p> Cases where the close operation may fail require careful
     * attention by implementers. It is strongly advised to relinquish
     * the underlying resources and to internally <em>mark</em> the
     * resource as closed, prior to throwing the exception. The {@code
     * close} method is unlikely to be invoked more than once and so
     * this ensures that the resources are released in a timely manner.
     * Furthermore it reduces problems that could arise when the resource
     * wraps, or is wrapped, by another resource.
     *
     * <p><em>Implementers of this interface are also strongly advised
     * to not have the {@code close} method throw {@link
     * InterruptedException}.</em>
     *
     * This exception interacts with a thread's interrupted status,
     * and runtime misbehavior is likely to occur if an {@code
     * InterruptedException} is {@linkplain Throwable#addSuppressed
     * suppressed}.
     *
     * More generally, if it would cause problems for an
     * exception to be suppressed, the {@code AutoCloseable.close}
     * method should not throw it.
     *
     * <p>Note that unlike the {@link java.io.Closeable#close close}
     * method of {@link java.io.Closeable}, this {@code close} method
     * is <em>not</em> required to be idempotent.  In other words,
     * calling this {@code close} method more than once may have some
     * visible side effect, unlike {@code Closeable.close} which is
     * required to have no effect if called more than once.
     *
     * However, implementers of this interface are strongly encouraged
     * to make their {@code close} methods idempotent.
     *
     * @throws Exception if this resource cannot be closed
     */
    void close() throws Exception;
}

그리고 전달할 자원은 반드시 final 또는 effectively final(자바 8 이상에서 도입된 개념으로, 변수가 사실상 final으로 동작하는 경우를 말함) 변수여야 합니다. try 블록이 실행되는 동안만 유효해야하기 때문입니다.

마지막으로 자원은 단일 문장으로 초기화되어야 합니다.

try (String path = "build.gradle";
     BufferedReader bufferedReader = new BufferedReader(new FileReader(path))) {
    return bufferedReader.readLine();
}

이런식의 소스 코드는 사용할 수 없고,

try (BufferedReader bufferedReader = new BufferedReader(new FileReader("build.gradle"))) {
    return bufferedReader.readLine();
}

이런식으로 한 문장으로 선언해야 합니다.

try-with-resources 구문을 사용하면 코드가 더 짧고 분명해지고 만들어지는 예외 정보가 훨씬 유용하다는 장점이 있습니다.

try-with-resources 구문을 사용했을 때 실제 코드는 어떻게 작성될까?

위에 작성했던 예시 코드

try (Resource resource = new Resource()) {
    // 자원 사용 코드
} catch (Exception e) {
    // 예외 처리 코드
}

를 바이트 코드로 변환하면 다음과 같습니다.

Resource resource = new Resource();
Throwable primaryException = null;
try {
    // 자원 사용 코드
} catch (Throwable e) {
    primaryException = e;
    throw e;
} finally {
    if (resource != null) {
        if (primaryException != null) {
            try {
                resource.close();
            } catch (Throwable suppressedException) {
                primaryException.addSuppressed(suppressedException);
            }
        } else {
            resource.close();
        }
    }
}

위의 변환된 바이트코드는 try 블록에서 자원을 사용하고, catch 블록에서 예외를 처리하며, finally 블록에서 자원을 해제하는 로직을 보여줍니다. finally 블록에서 자원을 해제하는 부분은 close() 메서드를 호출하여 자원을 해제하는 코드로 변환됩니다.

특이한 점은 첫 번째 예외가 발생했을 때 catch 블럭에서 미리 선언한 변수에 할당하고 다시 던져주고 finally 블럭에서 추가로 발생할 수 있는 예외를 첫 번째 예외의 suppressed로 등록해준다는 점입니다.

이렇게 처리하면 예외 로그가 보다 더 정확히 남는 것을 확인할 수 있습니다.

suppressed는 주 예외 처리 중에 발생한 부가적인 예외들을 의미합니다.

그래서 예외 정보가 어떻게 유용한 걸까?

예시 코드로 확인해보면,

class Resource implements AutoCloseable {
    @Override
    public void close() throws IOException {
        // 호출되면 내용이 서로 다른 예외를 발생시킴
        throw new IOException("Closing Resource Failed");
    }

    public void readData() throws IOException {
        // 호출되면 내용이 서로 다른 예외를 발생시킴
        throw new IOException("Reading Data Failed");
    }
}

public class TryWithResourceExceptionTest {
    public static void main(String[] args) throws IOException {
        try (Resource resource = new Resource()) {
            resource.readData();
        }
    }
}

이 예시를 실행하면 무조건 예외가 발생하게 되는데 다음과 같은 에러 로그를 확인할 수 있습니다.

Exception in thread "main" java.io.IOException: Reading Data Failed
    at Resource.readData(scratch_82.java:16)
    at TryWithResourceExceptionTest.main(scratch_82.java:23)
    Suppressed: java.io.IOException: Closing Resource Failed
        at Resource.close(scratch_82.java:12)
        at TryWithResourceExceptionTest.main(scratch_82.java:22)

예외가 발생한 순서대로 예외 로그를 표현하는데, 이후에 발생한 로그가 Suppressed에 명시된 것을 확인할 수 있습니다.

try-catch-finally에서 처리하게 되면 마지막에 발생한 예외만 표시해주기 때문에 정확히 어떤 부분에서 어떤 에러가 발생했는지 알기 어렵습니다. 특히 catch 블럭에서 상세한 예외를 처리하도록 해두었다면 그 외의 에러가 발생하여 제대로 수행되지 않았을 때도 해당 내용을 추적하기 어렵습니다. 이를 방지하기 위해 더 추상화된 레벨의 예외만 처리해도 마찬가지로 정확한 예외를 알기 어렵습니다.

 

한 줄 요약: try-catch-finally 대신 try-with-resources 씁시다!

당연히 try-with-resources에도 catch 블럭을 사용할 수 있습니다. 예시를 극단적으로 보여드리기위해 catch 블럭을 제외시켰으나 예외처리가 필요하다면 catch 블럭을 작성해야 합니다.