JAVA

[OOP] 제어자(modifier)

ewok 2023. 3. 7. 23:46

제어자

클래스, 변수, 메서드의 선언부에 함께 사용되어 부가적인 의미를 부여

접근 제어자    public, protected, default, private
그 외               static, final, abstract, native, transient, synchronized, volatile, strictfp

하나의 대상에 대해 여러 개의 제어자를 조합하여 사용할 수 있다.

단, 접근 제어자는 하나만 선택해서 사용할 수 있다.

 

 

static

'클래스의', '공통적인'이라는 의미를 가지고 있다.

 

static이 사용될 수 있는 곳 - 멤버변수, 메서드, 초기화 블록
대상 의미
멤버변수 - 모든 인스턴스에 공통적으로 사용되는 클래스변수가 된다.
- 클래스변수는 인스턴스를 생성하지 않고도 사용 가능하다.
- 클래스가 메모리에 로드될 때 생성된다.
메서드 - 인스턴스를 생성하지 않고도 호출이 가능한 static 메서드가 된다.
- static 메서드 내에서는 인스턴스 멤버들을 직접 사용할 수 없다.

 

final

'마지막의', '변경될 수 없는'의 의미를 가지고 있다.

final이 사용될 수 있는 곳 - 클래스, 메서드, 멤버변수, 지역변수
대상 의미
클래스 변경될 수 없는 크래스, 확장될 수 없는 클래스가 된다. 따라서 final로 지정된 클래스는 다른 클래스의 조상이 될 수 없다.
메서드 변경될 수 없는 메서드, final로 지정된 메서드는 오버라이딩을 통해 재정의 될 수 없다.
멤버변수 변수 앞에 final이 붙으면, 값을 변경할 수 없는 상수가 된다.
지역변수

 

생성자를 이용한 final 멤버 변수의 초기화

final이 붙은 변수는 상수이므로 보통은 선언과 초기화를 동시에 하지만, 인스턴스 변수의 경우 생성자에서 초기화할 수 있다.

class Card {
    final int NUMBER;		// 상수지만 선언과 함께 초기화하지 않고
    final String KIND;		// 생성자에서 단 한번만 초기화할 수 있다.

    Card(String kind, int num) {
        KIND = kind;
        NUMBER = num;
    }
}

 

 

abstract

'미완성'의 의미를 가지고 있다.

abstract가 사용될 수 있는 곳 - 클래스, 메서드
대상 의미
클래스 클래스 내에 추상 메서드가 선언되어 있음을 의미한다.
메서드 선언부만 작성하고 구현부는 작성하지 않은 추상 메서드임을 알린다.

추상 클래스는 아직 완성되지 않은 메서드가 존재하는 '미완성 설계도'이므로 인스턴스를 생성할 수 없다.

abstract class AbstractTest {	// 추상 클래스
	abstract void move();		// 추상 메서드
}

 

 

접근 제어자(access modifier)

  • 멤버 또는 클래스에 사용되어, 외부로부터의 접근을 제한한다.
  • 접근 제어자가 지정되어 있지 않다면 접근 제어자가 default이다.
접근 제어자가 사용될 수 있는 곳 - 클래스, 멤버변수, 메서드, 생성자\

private - 같은 클래스 내에서만 접근이 가능하다.
default - 같은 패키지 내에서만 접근이 가능하다.
protected - 같은 패키지 내에서, 그리고 다른 패키지의 자손 클래스에서 접근이 가능하다.
public - 접근 제한이 없다.
접근 범위 넓음     public > protected > default > private     접근 범위 좁음
대상 사용가능한 접근 제어자
클래스 public, default
메서드 public, protected, default, private
멤버변수
지역변수 없음

 

접근 제어자를 이용한 캡슐화

접근 제어자를 사용하는 이유는 클래스 내부에 선언된 데이터를 보호하기 위해서이다. 이것을 데이터 감추기라고 하며, 객체재향개념의 캡슐화에 해당하는 내용이다.

또 클래스 내에서만 사용되는, 내부 작업을 위해 임시로 사용되는 멤버변수나 부분작업을 처리하기 위한 메서드 등의 멤버들을 클래스 내부에 감추기 위해서이다. 이것 역시 캡슐화이다.

접근 제어자의 사용 이유
1. 외부로부터 데이터를 보호하기 위해
2. 외부에는 불필요한, 내부적으로만 사용되는 부분을 감추기 위해

 

 

생성자의 접근 제어자

  • 생성자에 접근 제어자를 사용함으로써 인스턴스의 생성을 제한할 수 있다.
  • 일반적으로 생성자의 접근 제어자는 클래스의 접근 제어자와 일치한다.
class Singleton {
    ...
    private static Singleton s = new Singleton();
    private Singleton() {
    	...
    }
    
    // 인스턴스를 생성하지 않고도 호출할 수 있어야 하므로 static이어야 한다.
    public static Singleton getInstance() {
    	return s;
    }
    ...
}

 

생성자가 private인 클래스는 다른 클래스의 조상이 될 수 없다. 자손 클래스의 인스턴스를 생성할 때 조상클래스의 생성자를 호출해야 하는데, 생성자의 접근 제어자가 private이므로 자손 클래스에서 호출하는 것이 불가능하기 때문이다. 따라서 클래스 앞에 final을 붙여 상속할 수 없는 클래스라는 것을 알리는 것이 좋다.

 

 

제어자의 조합

대상 사용가능한 제어자
클래스 public, default, final, abstract
메서드 모든 접근 제어자, final, abstarct, static
멤버변수 모든 접근 제어자, fianl, static
지역변수 final

 

제어자를 조합해서 사용할 때 주의사항

1. 메서드에 static과 abstarct를 함께 사용할 수 없다.

static메서드는 몸통이 있는 메서드에만 사용할 수 있기 때문이다.

 

2. 클래스에 abstarct와 final을 동시에 사용할 수 없다.

클래스에 사용되는 final은 클래스를 확장할 수 없다는 의미이고, abstract는 상속을 통해서 완성되어야 한다는 의미이므로 서로 모순된다.

 

3. abstract메서드의 접근 제어자가 private일 수 없다.

abstract메서드는 자손클래스에서 구현해주어야 하는데 접근 제어자가 private이면, 자손클래스에서 접근할 수 없기 때문이다.

 

4. 메서드에 private과 final을 같이 사용할 필요는 없다.

접근 제어자가 private인 메서드는 오버라이딩을 할 수 없기 때문이다. 이 둘 중 하나만 사용하면 된다.