-
클래스와 인터페이스 - 아이템 16. public 클래스에서는 public 필드가 아닌 접근자 메서드를 사용하라Study/Effective Java 2022. 12. 19. 11:29
public 클래스에서는 public 필드가 아닌 접근자 메서드를 사용하라
캡슐화의 이점을 제공하지 못하는 클래스
class Point { public int x; public int y; public Point(int x, int y) { this.x = x; this.y = y; } }
위와 같은 클래스는 데이터 필드에 직접 접근할 수 있으니, 캡슐화의 이점을 제공하지 못한다. API를 수정하지 않고는 내부 표현을 바꿀 수 없고, 불변식을 보장할 수 없으며, 외부에서 필드에 접근할 때 부수 작업을 수행할 수도 없다.
철저한 객체 지향 프로그래머는 이런 클래스를 상당히 싫어해서 필드들을 모두 private으로 바꾸고 public 접근자(getter)를 추가한다.
접근자와 변경자 메서드를 활용해 캡슐화한 클래스
public class Point { public int x; public int y; public Point(int x, int y) { this.x = x; this.y = y; } public int getX() { return x; } public int getY() { return y; } public void setX(int x) { this.x = x;} public void setY(int y) { this.y = y;} }
public 클래스에서라면 위와 같은 방식으로 선언해야 한다. 패키지 바깥에서 접근할 수 있는 클래스라면 접근자를 제공함으로써 클래스 내부 표현 방식을 언제든 바꿀 수 있는 유연성을 얻을 수 있다.
pacakge-private 클래스 혹은 private 중첩 클래스
public class TopPoint { private static class Point { public int x; public int y; public Point getPoint() { Point point = new Point(); point.x = 3; point.y = 5; return point; } } }
위와 같은 package-private 클래스 혹은 private 중첩 클래스라면 데이터 필드를 노출한다 해도 하등의 문제가 없다. 그 클래스가 표현하려는 추상 개념만 올바르게 표현해주면 된다. 이 방식은 클래스 선언 면에서나 이를 사용하는 클라이언트 코드 면에서나 접근자 방식보다 훨씬 깔끔하다.
클라이언트 코드가 이 클래스 내부 표현에 묶이기는 하나, 클라이언트도 어차피 이 클래스를 포함하는 패키지 안에서만 동작하는 코드일 뿐이다. 따라서, 패키지 바깥 코드는 전혀 손대지 않고도 데이터 표현 방식을 바꿀 수 있다.
자바 플랫폼 라이브러리에서 public 클래스의 필드를 직접 노출시킨 사례
1. java.awt.Point
public class Point extends Point2D implements java.io.Serializable { public int x; public int y; ... }
2. java.awt.Dimension 클래스
public class Dimension extends Dimension2D implements java.io.Serializable { public int width; public int height; ... }
자바 플랫폼 라이브러리에도 public 클래스의 필드를 직접 노출하지 말라는 규칙을 어기는 사례가 종종 있다. 대표적인 예가 java.awt.package 패키지의 Point와 Dimension 클래스다.
불변 필드를 노출한 public 클래스
public final class Time { private static final int HOURS_PER_DAY = 24; private static final int MINUTES_PER_HOUR = 60; public final int hour; public final int minute; public Time(int hour, int minute) { if (hour < 0 || hour > HOURS_PER_DAY) { throw new IllegalArgumentException("시간 : " + hour); } if (minute < 0 || minute > MINUTES_PER_HOUR) { throw new IllegalArgumentException("분 : " + minute); } this.hour = hour; this.minute = minute; } ... }
public 클래스의 필드가 불변이라면 직접 노출할 때의 단점이 조금은 줄어들지만, 여전히 좋은 생각은 아니다. API를 변경하지 않고는 표현 방식을 바꿀 수 없고, 필드를 읽을 때 부수 작업을 수행할 수 없다는 단점은 여전하다. 단, 불변식은 보장할 수 있게 된다.
최종 정리
public 클래스는 절대 가변 필드를 직접 노출해서는 안 된다. 불변 필드라면 노출해도 덜 위험하지만 완전히 안심할 수는 없다. 하지만, package-private 클래스나 private 중첩 클래스에서는 종종 필드(불변이든 가변이든)를 노출하는 편이 나을 때도 있다.
출처
ㆍ 이펙티브 자바 Effective Java 3/E. 조슈아 블로크 저자(글) · 개앞맵시(이복연) 번역
728x90'Study > Effective Java' 카테고리의 다른 글
클래스와 인터페이스 - 아이템 18. 상속보다는 컴포지션을 사용하라 (0) 2022.12.20 클래스와 인터페이스 - 아이템 17. 변경 가능성을 최소화하라 (0) 2022.12.19 클래스와 인터페이스 - 아이템 15. 클래스와 멤버의 접근 권한을 최소화하라 (0) 2022.12.16 모든 객체의 공통 메서드 - 아이템 14. Comparable을 구현할지 고려하라 (0) 2022.12.15 모든 객체의 공통 메서드 - 아이템 13. clone 재정의는 주의해서 진행하라 (0) 2022.12.15