이전 시간에 클래스 패스를 배웠다. 클래스 패스란 컴퓨터의 저장 장치 어딘가에 존재하는 클래스 파일을 사용하기 위한 방법이다. 이번 시간에 살펴볼 패키지(Package)는 하나의 클래스 안에서 같은 이름의 클래스들을 사용하기 위한 방법이라고 할 수 있다.
비유를 해보자. 서로 다른 내용의 파일 java.txt가 하나의 컴퓨터에 동시에 공존할 수는 없다. 그래서 고안된 것이 디렉토리다. java.txt 파일을 각각 a와 b라는 디렉토리에 저장한다면 하나의 컴퓨터 안에 같은 이름의 파일이 공존할 수 있게 된다. 누군가에게 'a 디렉토리에 있는 java.txt'를 이메일로 보내달라고 요청할 수 있게 되는 것이다.
패키지도 이와 유사하다. 클래스가 많아짐에 따라서 같은 이름을 가진 클래스가 생겨날 가능성이 높아지게 되는데 이름의 충돌을 방지하기 위한 고안된 것이 패키지라고 할 수 있다.
정보 공학에서는 '이름의 충돌'이라는 문제를 해결하기 위해서 다양한 노력을 하고 있다. 전역변수와 지역변수, 객체도 그런 연장선에 있다고 볼 수 있다.
이클립스에서 파일을 선택하고 오른쪽 클릭을 하고 메뉴 하단에 properties 항목을 선택하면 대화상자가 나타난다. Location이 소스코드가 위치하는 경로다. 생성한 프로젝트에서의 패키지와 Location 중간에 나타나는 부분이 같다는 것을 알 수 있다.
패키지는 기본적으로 디렉토리와 일치한다. 그렇기 때문에 아래의 패키지들은 물리적으로 같은 디렉토리에 존재할 수 없다.
- org.opentutorials.javatutorials.object
- org.opentutorials.javatutorials.classninstance
그럼 패키지는 실제로 어떻게 쓰이는가를 알아보자.
아래 코드를 보자. 아래 코드의 파일명은 A.java이다. 패키지명은 일반적으로 클래스를 제작한 개인이나 단체가 소속된 웹사이트의 도메인을 이용한다. 패키지의 이름도 중복될 수 있는데 웹사이트의 도메인은 전 세계에서 유일무일한 식별자이기 때문에 이러한 중복의 문제를 피할 수 있다.
package org.opentutorials.javatutorials.packages.example;
public class A {}
아래 코드는 위에서 정의한 클래스 A를 클래스 B에서 사용하는 예제다. 정상적으로 동작한다.
package org.opentutorials.javatutorials.packages.example;
public class B {
public static void main(String[] args) {
A a = new A();
}
}
이번에는 패키지를 바꿔보자.
package org.opentutorials.javatutorials.packages.example2;
public class B {
public static void main(String[] args) {
// 클래스 A가 다른 패키지에 있기 때문에 로드할 수 없다.
A a = new A();
}
}
위의 코드는 동작하지 않는다. 주석으로 처리한 A a = new A(); 부분에서 에러가 발생하기 때문이다. 그 이유는 여기서 사용하려는 클래스 A와 B가 서로 다른 패키지에 소속되어 있기 때문이다. 아래와 같이 코드를 고쳐서 이 문제를 해결할 수 있다.
package org.opentutorials.javatutorials.packages.example2;
import org.opentutorials.javatutorials.packages.example.A; // 패키지 import
public class B {
public static void main(String[] args) {
A a = new A();
}
}
서로 다른 패키지에 있는 클래스를 가져오려면 import를 통해서 다른 패키지의 클래스를 현재의 소스코드로 불러와야 한다. 만약 특정 패키지에 있는 모든 클래스를 로드하고 싶다면 아래와 같이 하면 된다.
package org.opentutorials.javatutorials.packages.example2;
import org.opentutorials.javatutorials.packages.example.*; // 모든 클래스 로드
public class B {
public static void main(String[] args) {
A a = new A();
}
}
손으로 컴파일을 해보자. 개발도구 없이 코딩하는 경우는 거의 없다. 그러므로 혹시 이해가 안 되면 나중에 다시 봐도 되는 부분이다. 하지만 언젠가는 알고 있어야 하는 부분이다. 이클립스 내부에서 어떤 일이 일어나는지 궁금하지 않은가?
프로젝트 디렉토리의 구성을 살펴보자.
- src : 소스 코드가 들어있다.
- bin : 컴파일된 클래스 파일이 들어있다.
위와 같이 구분한 이유는 관리의 편의성을 위해서다. 그럼 src에 소스코드를 만들고 그것을 컴파일 한 결과를 bin 하위에 위치하도록 작업해보자.
우선 우리가 컴파일하려는 클래스는 아래와 같은 패키지의 소속이다.
package org.opentutorials.javatutorials.packages.example3;
패키지는 디렉토리와 대응관계에 있기 때문에 패키지의 구조대로 디렉토리를 만들어보자.
packages 디렉토리에 파일 Selfcompile.java를 만든다. 파일의 내용은 아래와 같다.
package org.opentutorials.javatutorials.packages.example3;
public class Selfcompile{}
이제 컴파일을 해보자. 컴파일은 아래와 같이 프로젝트 디렉토리에서 진행하자. 컴파일을 하려면 콘솔을 실행시켜야 한다. 콘솔을 프로젝트 디렉토리로 이동한다. 이제 컴파일을 하자.
> javac src/org/opentutorials/javatutorials/packages/example3/*.java
위의 명령은 현재 디렉토리를 기준으로 src/org/opentutorials/javatutorials/packages/example3/ 하위에 있는 모든 자바 파일을 컴파일한다. 컴파일 한 결과는 src/org/opentutorials/javatutorials/packages/example3/ 에 저장된다. 특별한 옵션을 주지 않으면 소스코드와 클래스 파일이 동일한 디렉토리에 위치하게 된다.
우리가 원하는 것은 클래스 파일이 bin 하위에 위치하도록 하는 것이다. 아래와 같이 컴파일 명령을 바꾸면 된다.
> javac src/org/opentutorials/javatutorials/packages/example3/*.java -d bin
-d bin은 컴파일된 결과를 bin 디렉토리 하위에 위치시킨다는 의미다. 자바 컴파일러는 자동으로 클래스의 패키지에 해당하는 디렉토리를 생성해준다.
만약 import 한 패키지 안에 같은 이름의 클래스가 존재하고 이 클래스를 사용하고 싶다면 어떤 문제가 발생할까? 아래 코드는 import 하고 있는 두 개의 패키지에 클래스 B가 존재하는 경우에 어떤 일이 발생하는가를 보여준다.
package org.opentutorials.javatutorials.packages.example3;
import org.opentutorials.javatutorials.packages.example.*;
import org.opentutorials.javatutorials.packages.example2.*;
public class D {
public static void main(String[] args) {
B b = new B();
}
}
위의 코드는 오류가 발생한다. 클래스 B의 이름이 중복되기 때문에 애매함(ambiguous)의 문제가 발생한다. 아래와 같은 방법으로 이 문제를 우회할 수 있다.
package org.opentutorials.javatutorials.packages.example3;
import org.opentutorials.javatutorials.packages.example.*;
import org.opentutorials.javatutorials.packages.example2.*;
public class D {
public static void main(String[] args) {
org.opentutorials.javatutorials.packages.example2.B b = new org.opentutorials.javatutorials.packages.example2.B();
}
}