우리가 쇼핑몰 시스템을 개발한다고 가정해보자.
다음과 같이 프로그램이 매우 작고 단순해서 클래스가 몇개 없다면 크게 고민할 거리가 없겠지만, 기능이 점점 추가되어서 프로그램이 아주 커지게 된다면 어떻게 될까?
아주 작은 프로그램
Order
User
Product
큰 프로그램
User
UserManager
UserHistory
Product
ProductCatalog
ProductImage
Order
OrderService
OrderHistory
ShoppingCart
CartItem
Payment
PaymentHistory
Shipment
ShipmentTracker
매우 많은 클래스가 등장하면서 관련 있는 기능들을 분류해서 관리하고 싶을 것이다.
컴퓨터는 보통 파일을 분류하기 위해 폴더, 디렉토리라는 개념을 제공한다. 자바도 이런 개념을 제공하는데, 이것이 바로 패키지이다.
다음과 같은 카테고리를 만들고 분류해보자.
user
ㄴ User
ㄴ UserManager
ㄴ UserHistory
product
ㄴ Product
ㄴ ProductCataglog
ㄴ ProductImage
order
ㄴ Order
ㄴ OrderService
ㄴ OrderHistory
cart
ㄴ ShoppingCart
ㄴ CartItem
payment
ㄴ Payment
ㄴ PaymentHistory
shipping
ㄴ Shipment
ㄴ ShipmentTracker
여기서 user, product 등이 바로 패키지이다.
그리고 해당 패키지 안에 관련된 자바 클래스들을 넣으면 된다.
패키지(package)는 이름 그대로 물건을 운송하기 위한 포장 용기나 그 포장 묶음을 뜻한다.
패키지 사용법을 코드로 확인해보자.
패키지를 먼저 만들고 그 다음에 클래스를 만들어야 한다.
패키지 위치에 주의하자.
pack.Data
package pack;
public class Data {
public Data() {
System.out.println("패키지 pack Data 생성");
}
}
pack.a.User
package pack.a;
public class User {
public User() {
System.out.println("패키지 pack.a User 생성");
}
}
참고: 생성자에 public 을 사용했다. 다른 패키지에서 이 클래스의 생성자를 호출하려면 public 을 사용해야 한다. 자세한 내용은 접근 제어자에서 설명한다. 지금은 코드와 같이 생성자에 public 키워드를 넣어두자.
pack.PackageMain1
package pack;
public class PackageMain1 {
public static void main(String[] args) {
Data d = new Data();
pack.a.User u = new pack.a.User();
}
}
pack 패키지 위치에 PackageMain1 클래스를 만들었다.
실행 결과
패키지 pack Data 생성
패키지 pack.a User 생성
package pack;
import pack.a.User;
public class PackageMain2 {
public static void main(String[] args) {
Data d = new Data();
User u = new User();
}
}
코드에서 첫줄에는 package 를 사용하고, 다음 줄에는 import 를 사용할 수 있다.
import 를 사용하면 다른 패키지에 있는 클래스를 가져와서 사용할 수 있다.
import 를 사용한 덕분에 코드에서는 패키지 명을 생략하고 클래스 이름만 적을 수 있다.
여러개가 패키지에 있는 것을 전부 가져오려면
import pack.a.*
이런 식으로 전부 가져올 수 있다.
클래스 이름 중복
패키지 덕분에 클래스 이름이 같아도 패키지 이름으로 구분해서 같은 이름의 클래스를 사용할 수 있다.
pack.a.User
pack.b.User
이런 경우 클래스 이름이 둘다 User 이지만 패키지 이름으로 대상을 구분할 수 있다.
이렇게 이름이 같은 경우 둘다 사용하고 싶으면 어떻게 해야할까?
package pack.b;
public class User {
public User() {
System.out.println("패키지 pack.b User 생성");
}
}
package pack;
import pack.a.User;
public class PackageMain3 {
public static void main(String[] args) {
User userA = new User();
pack.b.User userB = new pack.b.User();
}
}
이런 식으로 직접 접근해야한다.
같은 이름의 클래스가 있다면 import 는 둘중 하나만 선택할 수 있다. 이때는 자주 사용하는 클래스를 import 하고 나머지를 패키지를 포함한 전체 경로로 적어주면 된다. 물론 둘다 전체 경로로 적어준다면 import 를 하지 않아도 된다.
이렇게 하면 다음과 같이 총 3개의 패키지가 존재한다.
a, a.b, a.c
계층 구조상 a 패키지 하위에 a.b 패키지와 a.c 패키지가 있다.
그런데 이것은 우리 눈에 보기에 계층 구조를 이룰 뿐이다. a 패키지와 a.b, a.c 패키지는 서로 완전히 다른 패키지이다.
따라서 a 패키지의 클래스에서 a.b 패키지가 필요하면 import 해서 사용해야 한다. 반대도 물론 마찬가지다.
정리하면 패키지가 계층 구조를 이루더라도 모든 패키지는 서로 다른 패키지이다.
물론 사람이 이해하기 쉽게 계층 구조를 잘 활용해서 패키지를 분류하는 것은 좋다. 참고로 카테고리는 보통 큰 분류에서 세세한 분류로 점점 나누어진다. 패키지도 마찬가지이다.
실제 패키지가 어떤 식으로 사용되는지 예제를 통해서 알아보자. 실제 동작하는 코드는 아니지만, 큰 애플리케이션은 대략 이런식으로 패키지를 구성한다고 이해하면 된다. 참고로 이것은 정답은 아니고 프로젝트 규모와 아키텍쳐에 따라서 달라진다.
전체 구조도
com.helloshop.user 패키지
package com.helloshop.user;
public class User {
String userId;
String name;
}
package com.helloshop.user;
public class UserService {
}
com.helloshop.product 패키지
package com.helloshop.product;
public class Product {
String productId;
int price;
}
package com.helloshop.product;
public class ProductService {
}
com.helloshop.order 패키지
package com.helloshop.order;
import com.helloshop.user.User;
import com.helloshop.product.Product;
public class Order {
User user;
Product product;
public Order(User user, Product product) {
this.user = user;
this.product = product;
}
}
package com.helloshop.order;
import com.helloshop.user.User;
import com.helloshop.product.Product;
public class OrderService {
public void order(){
User user = new User();
Product product = new Product();
Order order = new Order(user, product);
}
}
package com.helloshop.order;
public class OrderHistory {
}
패키지를 구성할 때 서로 관련된 클래스는 하나의 패키지에 모으고, 관련이 적은 클래스는 다른 패키지로 분리하는 것이 좋다.