Java8에서 final이 붙지 않은 변수의 값이 변경되지 않는다면, 그 변수를 Effectively final
이라고 합니다.
말보다 코드로 설명하는게 더 쉬울 것 같습니다.
다음 코드에서 변수 num
은 Effectively final입니다. num
은 선언과 동시에 10으로 할당되었고 객체가 소멸될 때까지 값이 변경되지 않았기 때문입니다.
int num = 10;
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println("number: " + num);
}
};
runnable.run();
위의 코드는 다음과 같이 Lamda expression으로 작성할 수도 있습니다.
int num = 10;
Runnable runnable = () -> System.out.println("number: " + num);
runnable.run();
Inner 클래스, 익명(anonymous) 클래스 내부에서 외부의 변수를 접근할 때는 final 변수만 접근이 가능합니다.
만약 위의 코드에서 변수 num
을 변경하면 어떻게 될까요?
다음 코드는 Lamda expression 안에서 num
을 변경하고 있습니다.
int num = 10;
Runnable runnable = () -> {
num++;
System.out.println("number: " + num);
};
runnable.run();
위 코드를 빌드하면, 다음과 같은 컴파일 에러가 발생합니다.
Error:(51, 17) java: local variables referenced from a lambda expression must be final or effectively final
에러 내용을 보면 final을 사용하거나 effectively final을 사용하라고 합니다.
즉, 값이 변경되는 변수 num
은 더 이상 effectively final이 아니라는 말입니다.
또한, 다음과 같이 Lambda expression 밖에서 값을 변경해도 동일한 컴파일 에러가 발생합니다.
int num = 10;
Runnable runnable = () -> {
System.out.println("number: " + num);
};
num++;
runnable.run();
정리
Inner 클래스, 익명(anonymous) 클래스 내부에서는 외부의 final 변수만 접근이 가능합니다. Java8에서 final이 붙지 않았지만, 값이 변하지 않는 변수를 Effectively final라고 하고, 이 변수들은 익명 클래스 내부에서 접근할 수 있습니다.
이로 인해, final을 생략할 수 있습니다.
개발자의 의도를 알 수 있게 명시적으로 final을 붙여주는 것이 좋을 것 같은데요, 요즘 non-nullable 타입을 제공하는 언어들도 많아서, final을 붙여주는 Java가 구식처럼 느껴집니다.
요즘 트렌드에 맞춰 추가된 기능이 아닌가 싶습니다.
참고
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 가져오기