try-with-resources
는 try(...)
에서 선언된 객체들에 대해서 try가 종료될 때 자동으로 자원을 해제해주는 기능입니다.
try에서 선언된 객체가 AutoCloseable을 구현하였다면 Java는 try구문이 종료될 때 객체의 close()
메소드를 호출해 줍니다.
자바6에서 리소스 사용 및 해제하는 방법을 알아보고, try-with-resources로 동일한 코드를 리팩토링해보면서 장점이 무엇인지 알아보겠습니다.
try-catch-finally로 자원 해제
Java7 이전에, try-catch-finally 구문에서 자원을 해제하려면 정말 귀찮았고 코드 양도 많고 매우 지저분했습니다.
예를 들어, 다음 코드는 try-catch-finally
을 사용하여 파일을 열고 문자열을 모두 출력하는 코드입니다.
public static void main(String args[]) throws IOException {
FileInputStream is = null;
BufferedInputStream bis = null;
try {
is = new FileInputStream("file.txt");
bis = new BufferedInputStream(is);
int data = -1;
while((data = bis.read()) != -1){
System.out.print((char) data);
}
} finally {
// close resources
if (is != null) is.close();
if (bis != null) bis.close();
}
}
코드를 보시면 try에서 InputStream 객체를 생성하고 finally에서 close를 해주었습니다. try 안의 코드를 실행하다 Exception이 발생하는 경우 모든 코드가 실행되지 않을 수 있기 때문에 finally에 close 코드를 넣어주어야 합니다. 심지어 InputStream 객체가 null인지 체크해줘야 하며 close에 대한 Exception 처리도 해야 합니다. 저는 main에서 IOException를 throws한다고 명시적으로 선언했기 때문에 close에 대한 try-catch 구문을 작성하지 않았습니다.
Try-with-resources로 자원 쉽게 해제
Java7부터 Try-with-resources 구문을 지원하고 이것을 사용하면 자원을 쉽게 해제할 수 있습니다.
다음 코드는 Try-with-resources
를 사용하여 InputStream으로 파일의 문자열을 모두 출력하는 코드입니다. 실행 결과는 위의 예제와 동일합니다.
public static void main(String args[]) {
try (
FileInputStream is = new FileInputStream("file.txt");
BufferedInputStream bis = new BufferedInputStream(is)
) {
int data = -1;
while ((data = bis.read()) != -1) {
System.out.print((char) data);
}
} catch (IOException e) {
e.printStackTrace();
}
}
코드를 보시면, try(...)
안에 InputStream 객체 선언 및 할당하였습니다. 여기에서 선언한 변수들은 try 안에서 사용할 수 있습니다.
코드의 실행 위치가 try 문을 벗어나면 try-with-resources는 try(...)
안에서 선언된 객체의 close()
메소드들을 호출합니다.
그래서 finally에서 close()
를 명시적으로 호출해줄 필요가 없습니다.
try-with-resources에서 자동으로 close가 호출되는 것은 AutoCloseable을 구현한 객체에만 해당이 됩니다. 이 부분은 아래에서 좀 더 자세히 설명하겠습니다.
try-with-resources의 장점은 코드를 짧고 간결하게 만들어 읽기 쉽고 유지보수가 쉬워집니다. 또한 명시적으로 close를 호출하려면 많은 if와 try-catch를 사용해야 하기 때문에 실수로 close를 빼먹는 경우가 있습니다. 이것을 이용하면 이런 자잘한 버그들이 발생할 가능성이 적습니다.
Try-with-resources로 close()가 호출되는 객체는?
Try-with-resources가 모든 객체의 close()
를 호출해주지는 않습니다. AutoCloseable을 구현한 객체만 close()
가 호출됩니다.
AutoCloseable은 인터페이스이며 자바7부터 지원합니다.
package java.lang;
public interface AutoCloseable {
void close() throws Exception;
}
위의 예제에서 BufferedInputStream의 상속구조는 다음과 같습니다.
java.lang.Object
java.io.InputStream
java.io.FilterInputStream
java.io.BufferedInputStream
InputStream은 AutoCloseable를 상속받은 Closeable을 구현하였습니다.
public abstract class InputStream extends Object implements Closeable {
....
}
public interface Closeable extends AutoCloseable {
void close() throws IOException;
}
이런 이유로 위의 예제에서 BufferedInputStream 객체가 try-with-resources에 의해서 해제될 수 있었습니다.
AutoCloseable을 구현하는 클래스 만들기
내가 만든 클래스가 try-with-resources으로 자원이 해제되길 원한다면 AutoCloseable을 implements해야 합니다.
아래 코드에서 CustomResource 클래스는 AutoCloseable을 구현하였습니다. main에서는 이 객체를 try-with-resources로 사용하고 있습니다.
public static void main(String args[]) {
try (CustomResource cr = new CustomResource()) {
cr.doSomething();
} catch (Exception e) {
}
}
private static class CustomResource implements AutoCloseable {
public void doSomething() {
System.out.println("Do something...");
}
@Override
public void close() throws Exception {
System.out.println("CustomResource.close() is called");
}
}
실행해보면 다음과 같이 출력됩니다.
이 예제에서는 close()
가 호출될 때 로그를 출력하기 때문에 close()
가 실제로 호출되는지 눈으로 확인할 수 있습니다.
Do something...
CustomResource.close() is called
참고
Related Posts
- Java - Unsupported class file major version 61 에러
- Java - String.matches()로 문자열 패턴 확인 및 다양한 예제 소개
- Java - 문자열 공백제거 (trim, replace)
- Java - replace()와 replaceAll()의 차이점
- Java - ArrayList 초기화, 4가지 방법
- Java - 배열 정렬(Sorting) (오름차순, 내림차순)
- Java - 문자열(String)을 비교하는 방법 (==, equals, compare)
- Java - StringBuilder 사용 방법, 예제
- Java - 로그 출력, 파일 저장 방법 (Logger 라이브러리)
- Java IllegalArgumentException 의미, 발생 이유
- Java - NullPointerException 원인, 해결 방법
- Seleninum의 ConnectionFailedException: Unable to establish websocket connection 해결
- Java - compareTo(), 객체 크기 비교
- Java - BufferedWriter로 파일 쓰기
- Java - BufferedReader로 파일 읽기
- Java charAt() 함수 알아보기
- Java - BigInteger 범위, 비교, 연산, 형변환
- Java contains()로 문자(대소문자 X) 포함 확인
- Java - Set(HashSet)를 배열로 변환
- Java - 문자열 첫번째 문자, 마지막 문자 확인
- Java - 문자열 한글자씩 자르기
- Java - 문자열 단어 개수 가져오기
- Java - 1초마다 반복 실행
- Java - 배열을 Set(HashSet)로 변환
- Java - 여러 Set(HashSet) 합치기
- Java - 명령행 인자 입력 받기
- Java - 리스트 역순으로 순회, 3가지 방법
- Java - 특정 조건으로 리스트 필터링, 3가지 방법
- Java - HashMap 모든 요소들의 합계, 평균 계산
- Java - 특정 조건으로 HashMap 필터링
- Java - 싱글톤(Singleton) 패턴 구현
- Java - 숫자 왼쪽에 0으로 채우기
- Java - String 배열 초기화 방법
- Java - 정렬된 순서로 Map(HashMap) 순회
- Java - HashMap에서 key, value 가져오기