맴버(member)는 영어로 구성원이라는 뜻이다. 객체도 구성원이 있는데 아래와 같다.
- 변수
- 메소드
객체를 만들기 위해서 우선 클래스를 정의하고, 클래스에 대한 인스턴스를 만들었다. "18. 클래스와 인스턴스 그리고 객체"에서 살펴봤던 예제 CalculatorDemo에서 left와 right 변수는 누구의 맴버일까?
인스턴스의 맴버다. 인스턴스를 만들어야 사용할 수 있었고, 인스턴스마다 서로 다른 값을 가지고 있기 때문이다.
그렇다면 클래스도 맴버를 가질 수 있다는 것일까? 클래스도 맴버를 가질 수 있다.
CalculatorDemo에서 사용한 인스턴스 변수인 left를 놓고 생각해보자. left의 값은 인스턴스마다 달라질 수 있다. 인스턴스 변수 c1의 left 값은 10이고, c2의 left 값은 20이었다. 인스턴스의 상태인 변수의 값이 인스턴스마다 다른 값을 가질 수 있다는 점은 하나의 클래스를 여러 개의 인스턴스로 만들어서 사용할 수 있다는 점에서 좋은 기능이라고 할 수 있다. 그런데 경우에 따라서 모든 인스턴스가 같은 값을 공유하게 하고 싶을 때가 있다.
예를 들면, 우리가 만든 계산기가 원주율의 값을 사용자에게 제공하는 것을 생각해보자. 그런데 원주율인 3.14는 이미 알려져있는 수이다. 따라서 각각의 인스턴스마다 원주율의 값을 별도로 가지고 있을 필요가 없다. 이런 경우 변수를 클래스의 맴버로 만들면 된다. 아래 코드는 원주율을 담고 있는 변수 PI를 클래스의 소속인 맴버로 만든 예제다.
class Calculator {
static double PI = 3.14;
int left, right;
public void setOprands(int left, int right) {
this.left = left;
this.right = right;
}
public void sum() {
System.out.println(this.left + this.right);
}
public void avg() {
System.out.println((this.left + this.right) / 2);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
Calculator c1 = new Calculator();
System.out.println(c1.PI);
Calculator c2 = new Calculator();
System.out.println(c2.PI);
System.out.println(Calculator.PI);
}
}
변수 PI의 앞에 static이 붙었다. static을 맴버(변수,메소드) 앞에 붙이면 클래스의 맴버가 된다. 클래스 소속의 변수를 만드는 법을 알았으니, 이번에는 이를 사용하는 방법에 대해 알아보자. 아래는 클래스 변수에 접근하는 두 가지 방법을 보여준다.
System.out.println(c1.PI); // 인스턴스를 통해서 PI에 접근
System.out.println(Calculator.PI); // 클래스를 통해서 PI에 접근
두 번째 방법은 원주율만 필요할 때 클래스에 직접 접근하기 때문에 인스턴스를 생성할 필요가 없다.
클래스 변수는 변수의 변경사항을 모든 인스턴스에서 공유해야 할 때도 사용한다. 만약 계산을 할 때 특별한 값을 포함시켜야 한다면 어떻게 해야 할까? 아래 예제는 sum과 avg를 실행할 때마다 특정한 값을 연산에 포함시키고 싶을 때 시도해볼 수 있는 방법이다.
class Calculator {
static double PI = 3.14; // 클래스 변수
static int base = 0; // 클래스 변수
int left, right;
public void setNum(int left, int right) {
this.left = left;
this.right = right;
}
public void sum() {
System.out.println(this.left + this.right + base); // base 값 포함
}
public void avg() {
System.out.println((this.left + this.right + base) / 2); // base 값 포함
}
}
public class CalculatorDemo {
public static void main(String[] args) {
Calculator c1 = new Calculator();
c1.setNum(10, 20); // 30
c1.sum();
Calculator c2 = new Calculator();
c2.setNum(20, 40); // 60
c2.sum();
// 클래스 변수 base의 값을 10으로 지정했다.
Calculator.base = 10;
c1.sum(); // 40
c2.sum(); // 70
}
}
클래스 변수 base 값을 변경한 결과 모든 인스턴스의 base 값이 모두 변경되었다.
클래스 변수의 용도를 정리해보면 아래와 같다.
지금까지 클래스 변수에 대해서 알아봤다. 클래스 변수가 있다면 클래스 메소드도 있지 않을까? 물론 있다.
예제 Calculator는 인스턴스 변수 left와 right를 이용해서 합계(sum)과 평균(avg)을 계산한다. 생각해보면 굳이 인스턴스가 left와 right의 값을 항상 유지하고 있어야 할 이유는 없다. 합계나 평균을 구할 때마다 좌항과 우항의 값을 주는 방식으로 계산할 수도 있다. 아래 예제를 보자.
class Calculator {
public static void sum(int left, int right){
System.out.println(left+right);
}
public static void avg(int left, int right){
System.out.println((left+right)/2);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
Calculator.sum(10, 20);
Calculator.avg(10, 20);
Calculator.sum(20, 40);
Calculator.avg(20, 40);
}
}
만약 메소드가 인스턴스 변수를 참조하지 않는다면 클래스 메소드를 사용해서 불필요한 인스턴스의 생성을 막을 수 있다.
아래 예제는 클래스와 인스턴스의 차이점을 보여주기 위한 예제다. 이 예제는 오류가 포함되어 있기 때문에 실행되지 않을 것이다. 예제의 내용을 살펴보기 전에 몇 가지 원칙을 기억해 둔다면 이 예제를 이해하는 것이 조금 수월할 것이다.
- 인스턴스 메소드는 클래스 맴버에 접근 할 수 있다.
- 클래스 메소드는 인스턴스 맴버에 접근 할 수 없다.
인스턴스 변수는 인스턴스가 만들어지면서 생성되는데, 클래스 메소드는 인스턴스가 생성되기 전에 만들어지기 때문에 클래스 메소드가 인스턴스 맴버에 접근하는 것은 존재하지 않는 인스턴스 변수에 접근하는 것과 같다
class C1 {
static int static_variable = 1;
int instance_variable = 2;
/* 클래스 메소드에서는 클래스 변수에 접근 가능 */
static void static_static(){
System.out.println(static_variable);
}
/* 클래스 메소드에서는 인스턴스 변수에 접근 불가 */
static void static_instance(){
System.out.println(instance_variable); // ERROR
}
/* 인스턴스 메소드에서는 클래스 변수에 접근 가능 */
void instance_static(){
System.out.println(static_variable);
}
/* 인스턴스 메소드에서는 인스턴스 변수에 접근 가능 */
void instance_instance(){
System.out.println(instance_variable);
}
}
public class ClassMemberDemo {
public static void main(String[] args) {
C1 c = new C1(); // 인스턴스 생성
/* 인스턴스로 접근 */
c.static_static(); // O
c.static_instance(); // ERROR → 클래스 메소드에는 접근 가능하지만, 인스턴스 변수에 접근 불가
c.instance_static(); // O
c.instance_instance(); // O
/* 클래스로 접근 */
C1.static_static(); // O
C1.static_instance(); // ERROR
C1.instance_static(); // ERROR
C1.instance_instance(); // ERROR
}
}
인스턴스 변수와 클래스 변수는 아래와 같이 부르기도 한다.
인스턴스 변수 -> Non-Static Field
클래스 변수 -> Static Field
필드(field)라는 것은 클래스 전역에서 접근 할 수 있는 변수를 의미하는데 이에 대한 자세한 내용은 유효범위 수업에서 알아보겠다.