지역변수의 유효 범위를 최소로 줄이면 코드 가독성과 유지보수성이 높아지고 오류 가능성은 낮아진다.
지역변수의 범위를 줄이는 방법
1. 가장 처음 쓰일 때 선언하기
- 사용하기 전에 미리 선언부터 해두면 코드가 어수선해져 가독성이 떨어진다.
- 변수를 실제로 사용하는 시점에 타입과 초깃값이 기억나지 않을 수 있다.
지역변수를 생각 없이 선언하다 보면 변수가 쓰이는 범위보다 너무 앞서 선언하거나, 다 쓴 뒤에도 여전히 살아 있게 되기 쉽다.
지역변수의 범위는 선언된 지점부터 그 지점을 포함한 블록이 끝날 때까지 이므로, 실제 사용하는 블록 바깥에 선언된 변수는 그 블록이 끝난 뒤까지 살아있게 된다.
2. 거의 모든 지역변수는 선언과 동시에 초기화해야 한다.
- 초기화에 필요한 정보가 충분하지 않다면 충분해질 때까지 선언을 미뤄야 한다.
- try-catch 문은 이 규칙에서 예외다.
- 변수 값을 try 블록 바깥에서도 사용해야 한다면 try 블록 앞에서 선언해야 한다.
- 변수를 초기화하는 표현식에서 검사 예외를 던질 가능성이 있다면 try 블록 안에서 초기화해야 한다.
- 그렇지 않으면 예외가 블록을 넘어 메서드에까지 전파된다.
try 블록 안에서 초기화
3. 메서드를 작게 유지하고 한가지 기능에 집중하자.
- 한 메서드에서 여러가지 기능을 처리한다면 그중 한 기능과만 관련된 지역변수라도 다른 기능을 수행하는 코드에서 접근할 수 있다.
- 메서드를 기능별로 쪼개서 이를 해결하자.
4. 반복문 관용구
for(int i=0, n=expensiveComputation(); i<n; i++){
//...i로 무언가를 한다.
}
- 범위가 정확히 일치하는 두 반복 변수 i와 n
- 반복 여부를 결정짓는 변수 i의 한곗값을 변수 n에 저장하여, 반복 때마다 다시 계산해야 하는 비용을 없앴다.
- 같은 값을 반환하는 메서드를 매번 호출한다면 이 관용구를 사용하자.
반복문의 변수 범위
- 반복문은 독특한 방식으로 변수 범위를 최소화 해준다.
- for형태나 for-each 형태의 반복문에서 반복 변수의 범위가 반복문의 몸체, 그리고 for 키워드와 몸체 사이의 괄호 안으로 제한된다.
- 따라서 반복 변수의 값을 반복문이 종료된 뒤에도 써야 하는 상황이 아니라면 while 문보다는 for문을 쓰는 게 낫다.
컬렉션을 순회할 때 권장하는 관용구
//for-each문
for(Element e : c){
//..e로 무언가를 한다.
}
//for문
for(Iterator<Element> i = c.iterator(); i.hasNext();){
Element e = i.next();
//...e와 i로 무언가를 한다.
}
- 반복자를 사용해야 하는 상황(반복자의 remove 메서드를 사용하는 경우 등)이면 for-each 문 대신 전통적인 for문을 사용하는 것이 낫다.
while문 - 복사 붙여 넣기 오류
//while문
Iterator<Element> i = c.iterator();
while(i.hasNext()) {
doSomethis(i.next());
}
Iterator<Element> i2 = c.iterator();
while(i.hasNext()) { //버그!
doSomethis(i2.next());
}
- 두 번째 while문에는 복사해 붙여넣기 오류가 있다.
- 새로운 반복 변수 i2를 초기화했지만, 실수로 이전 while문의 i를 다시 썼는데, i의 유효범위가 끝나지 않았으므로 코드는 컴파일도 잘되고 예외도 던지지 않는다.
- 또한 두 번째 while문은 c2를 순회하지 않고 끝나 버린다.
for문 - 복사 붙여 넣기 오류
for(Iterator<Element> i = c.iterator(); i.hasNext();){
Element e = i.next();
//...e와 i로 무언가를 한다.
}
//다음 코드는 "i를 찾을 수 없다"는 컴파일 오류를 낸다.
for (Iterator<Element> i2 = c2.iterator(); i.hasNext();){
Element e2 = i2.next();
//...e와 i2로 무언가를 한다.
}
- for문을 사용하면 이런 복사 붙여넣기 오류를 컴파일 타임에 잡아준다.
- 첫 번째 반복문이 사용한 원소와 반복자의 유효 범위가 반복문 종료와 함께 끝나기 때문이다.
- 변수 유효 범위가 for문 범위와 일치하여 똑같은 이름의 변수를 여러 반복문에서 써도 서로 아무런 영향을 주지 않는다.
- while문보다 짧아서 가독성이 좋다.
'Java > Effective Java' 카테고리의 다른 글
[Effective Java] Item 59. 라이브러리를 익히고 사용하라 (0) | 2023.07.23 |
---|---|
[Effective Java] Item 58. 전통적인 for 문보다는 for-each 문을 사용하라 (0) | 2023.07.23 |
[Effective Java] Item 56. 공개된 API 요소에는 항상 문서화 주석을 작성하라 (0) | 2023.07.22 |
[Effective Java] Item 55. 옵셔널 반환은 신중히 하라 (0) | 2023.07.22 |
[Effective Java] Item 54. null이 아닌, 빈 컬렉션이나 배열을 반환하라 (0) | 2023.07.16 |