맞는데 왜 틀릴까..?

Java/Effective Java

[Effective Java] Item 57. 지역변수의 범위를 최소화하라

안도일 2023. 7. 22. 19:10

지역변수의 유효 범위를 최소로 줄이면 코드 가독성과 유지보수성이 높아지고 오류 가능성은 낮아진다.

 

지역변수의 범위를 줄이는 방법

 

 

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문보다 짧아서 가독성이 좋다.