Java/Effective Java 3E

[이펙티브자바 3판] ITEM58. 전통적인 for 보다는 for-each 를 사용하라

잭피 2021. 2. 4. 00:02

이번장의 핵심은...

전통적인 for 문과 비교했을 때 for-each 문은 명료하고, 유연하고, 버그를 예방해줍니다

성능 저하도 없습니다

가능한 모든 곳에서 for 문이 아닌 for-each 문을 사용합시다


for vs for-each

전통적인 for문으로 컬렉션 순회하는 방법

// 컬렉션 순회 - 더 나은 방법이 존재
for (Iterator<Element> i = c.iterator(); i.hasNext(); ) {
	Element e = i.next();
	... // e로 무언가를 한다
}
// 배열 순회 - 더 나은 방법이 있음
for (int i=0; i<a.length; i++) {
		... // a[i]로 무언가를 한다
	}

→ while 문보다는 낫지만 가장 좋은 방법은 아닙니다

반복자와 인덱스 변수는 모두 코드를 지저분하게 합니다 (진짜 필요한건 원소)

 

for-each문으로 컬렉션 순회하는 방법

(for-each 문의 정식 이름은 향상된 for 문입니다)

for (Element e : elements) {
	... // e로 무언가를 한다
}

반복 대상이 컬렉션이든 배열이든, for-each 문을 사용해도 속도는 같습니다

컬렉션을 중첩해 순회해야 한다면 for-each 문의 이점이 더욱 커집니다

for (Suit suit : suits)
	for (Rank rank : ranks)
		deck.add(new Card(suit, rank));

for-each 문을 사용할 수 없는 상황 3가지

1. 파괴적인 필터링

컬렉션을 순회하면서 선택된 원소를 제거해야 한다면 반복자의 remove 메서드를 호출해야 합니다

자바 8부턴 Collection의 removeIf 메서드를 사용해 컬렉션의 명시적으로 순회하는 일을 피할 수 있습니다

 

2. 변형

리스트나 배열을 순회하면서 그 원소의 값 일부 혹은 전체를 교체해야 한다면 리스트의 반복자나 배열의 인덱스를 사용해야 합니다

 

3. 병렬 반복

여러 컬렉션을 병렬로 순회해야 한다면 각각의 반복자와 인덱스 변수를 사용해 엄격하고 명시적으로 제어해야 합니다

 

→ 3가지 상황 중 하나에 속할 때는 일반적인 for 문을 사용합시다

 

for-each 문은 컬렉션과 배열은 물론 Iterable 인터페이스를 구현한 객체라면 무엇이든 순회할 수 있습니다

public interface Iterable<E> {
	Iterator<E> iterator();
}

Iterable을 처음부터 직접 구현하기는 까다롭지만, 원소들의 묶음을 표현하는 타입을 작성해야 한다면 Iterable을 구현하는 쪽으로 고민해봅시다 (해당 타입을 Collection 인터페이스는 구현하지 않기로 했더라도)

구현해두면 그 타입을 사용하는 프로그래머는 for-each 문을 사용할 때마다 감사해할 것입니다