Java 기본 과정

클래스 맴버와 인스턴스 맴버

객체의 맴버
맴버(member)는 영어로 구성원이라는 뜻이다. 객체도 구성원이 있는데 아래와 같다.
o    변수
o    메소드
객체를 만들기 위해서 우선 클래스를 정의하고, 클래스에 대한 인스턴스를 만들었다. 복습을 해보자. 이전 시간에 살펴봤던 예제 CalculatorDemo.java에서 left와 right 변수는 누구의 맴버일까? 인스턴스의 맴버다. 인스턴스를 만들어야 사용할 수 있었고, 인스턴스마다 서로 다른 값을 가지고 있기 때문이다. 
그렇다면 클래스도 맴버를 가질 수 있다는 것일까? 아래 그림과 같이 클래스도 맴버를 가질 수 있다. 그 방법을 알아보는 것이 이번 수업의 목표다. 
 
클래스 변수
CalculatorDemo.java에서 사용한 인스턴스 변수인 left를 놓고 생각해보자. left의 값은 인스턴스마다 달라질 수 있다. 그런데 경우에 따라서 모든 인스턴스가 같은 값을 공유하게 하고 싶을 때가 있다.
이를테면 우리가 만든 계산기가 원주율의 값을 사용자에게 제공하도록 하고 싶다고 간주해보자. 그런데 원주율인 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 CalculatorDemo1 {
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);
}
}
5행을 보자.
static double PI = 3.14;
 
변수 PI의 앞에 static이 붙었다. static을 맴버(변수,메소드) 앞에 붙이면 클래스의 맴버가 된다. 
클래스 소속의 변수를 만드는 법을 알았으니까 이번에는 이것을 사용하는 법을 알아보자. 아래는 클래스 변수에 접근하는 방법 두가지를 보여준다.
 
// 1.인스턴스를 통해서 PI에 접근
System.out.println(c1.PI);
// 2. 클래스를 통해서 PI에 접근
System.out.println(Calculator.PI);
 
두번째 방법은 객체 Calculator.java의 다른 기능(sum, avg)은 필요없고, 원주율만 필요할 때 클래스에 직접 접근하기 때문에 인스턴스를 생성할 필요가 없어진다.
 
클래스 변수의 용도를 정리해보면 아래와 같다.
o    인스턴스에 따라서 변하지 않는 값이 필요한 경우 (이런 경우 final을 이용해서 상수로 선언하는 것이 바람직하다)
o    인스턴스를 생성할 필요가 없는 값을 클래스에 저장하고 싶은 경우
o    값의 변경 사항을 모든 인스턴스가 공유해야 하는 경우
 
클래스 메소드
지금까지 클래스 변수에 대해서 알아봤다. 클래스 변수가 있다면 클래스 메소드도 있지 않을까? 물론 있다.
예제 Calculator은 인스턴스 변수 left와 right를 이용해서 합계(sum)과 평균(avg)을 계산한다. 생각해보면 굳이 인스턴스가 left와 right의 값을 항상 유지하고 있어야 할 이유는 없다. 합계나 평균을 구할 때마다 좌항과 우항의 값을 주는 방식으로 계산을 할 수도 있다.  
 
class Calculator3{
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 CalculatorDemo3 {
public static void main(String[] args) {
  Calculator3.sum(10, 20);
  Calculator3.avg(10, 20);
  Calculator3.sum(20, 40);
  Calculator3.avg(20, 40);
}
}
만약 메소드가 인스턴스 변수를 참조하지 않는다면 클래스 메소드를 사용해서 불필요한 인스턴스의 생성을 막을 수 있다.
 
클래스 맴버와 인스턴스 맴버는 아래와 같은 원칙이 있다. 
1.    인스턴스 메소드는 클래스 맴버에 접근 할 수 있다.
2.    클래스 메소드는 인스턴스 맴버에 접근 할 수 없다. 
 
인스턴스 변수는 인스턴스가 만들어지면서 생성되는데, 클래스 메소드는 인스턴스가 생성되기 전에 만들어지기 때문에 클래스 메소드가 인스턴스 맴버에 접근하는 것은 존재하지 않는 인스턴스 변수에 접근하는 것과 같다.
 
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);
}
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();
// 인스턴스를 이용해서 정적 메소드에 접근 -> 성공
c.static_instance();
// 인스턴스를 이용해서 인스턴스 메소드에 접근 -> 성공
c.instance_static();
// 인스턴스를 이용해서 인스턴스 메소드에 접근 -> 성공 
c.instance_instance();
 
// 클래스를 이용해서 클래스 메소드에 접근 -> 성공
C1.static_static();
// 클래스를 이용해서 클래스 메소드에 접근 -> 성공
C1.static_instance();
// 클래스를 이용해서 인스턴스 메소드에 접근 -> 실패
//C1.instance_static();
// 클래스를 이용해서 인스턴스 메소드에 접근 -> 실패
//C1.instance_instance();
}
}
 
인스턴스 변수와 클래스 변수는 아래와 같이 부르기도 한다.
o    인스턴스 변수 -> Non-Static Field
o    클래스 변수 -> Static Field
 
 

댓글

댓글 본문