오버라이딩(Override)
기각하다, 무시하다의 뜻을 담고 있다.
상속 관계에 있는 클래스 간에 하위클래스가 상위 클래스와 완전 동일한 메소드를 덮어 쓴다는 의미
참조변수를 통해 인스턴스의 메소드를 호출하면 하위클래스의 메소드가 호출
하위 클래스에서 오버라이딩 된 상위 클래스의 메소드를 호출하려면 super를 사용
참조변수로 인스턴스를 참조하면, 실제로 참조하는 인스턴스 종류에 상관없이 상위 클래스에 정의된
메소드만 호출 가능하다.
참조변수를 통해 정의되어 있는 메소드만 호출 가능하도록 제한하는 방식으로 컴파일러가 디자인 되어있다.
상속의 참조관계
class First {...}
class Second extends First {...}
class Third extends Second {...}
위 코드는 문제없이 컴파일이 된다. 상위 클래스의 참조변수로도 하위 클래스의 인스턴스 참조가 가능하기 때문이다.
전체적으로 Third는 Second를 상속하고 연쇄적으로 Second는 First를 상속한다.
따라서 Third도 First를 당연히 상속한다.
Third ref1 = ... // compile success
Second ref2 = ref1; // Third is a Second
First ref3 = ref2; // Third is a First
여기서 Third ref1이 무엇이 되었든간에 컴파일에 성공했다면, 그 다음행은 정상적으로 실행된다.
왜냐하면 Third는 Second이기도 하며, Third는 First이기도 하므로 참조가 가능하기 때문이다.
First ref1 = new Third(); // 컴파일러가 ref1을 First 타입으로 판단
Second ref2 = ref1; // First is not a Second, compile error!!
Third ref3 = ref1; // Fisrt is not a Third, compile error!!
First ref1 = new Third(); // 컴파일러가 ref1을 First 타입으로 판단
Second ref2 = new First(); // First is not a Second, compile error!!
Third ref3 = new First(); // Fisrt is not a Third, compile error!!
반면에 위 코드는 컴파일 에러가 발생한다.
First타입의 참조변수 ref1에 Third의 인스턴스를 담지만 컴파일러는 ref1을 First타입으로 판단하므로, 나머지 행에서 하위 클래스인 Second와 Third의 참조변수에 상위 클래스인 First타입의 참조값을 담으려고 하자 컴파일 에러가 발생한다.
오버라이딩(Override)과 오버로딩(Overload)
class GrandParents {
public void rideMethod() { System.out.println("GrandParants' rideMethod"); }
public void loadMethod() { System.out.println("GrandParants' loadMethod"); }
}
class Parents extends GrandParents {
public void rideMethod() { System.out.println("Parants' rideMethod"); }
public void loadMethod(int n) { System.out.println("Parants' loadMethod"); }
}
class Child extends Parents {
public void rideMethod() { System.out.println("Child's rideMethod"); }
public void loadMethod(double n) { System.out.println("Child's loadMethod"); }
}
class RideLoad {
public static void main(String[] args) {
GrandParents ref1 = new Child();
Parents ref2 = new Child();
Child ref3 = new Child();
ref1.rideMethod(); // Child.rideMethod() 호출
ref2.rideMethod(); // Child.rideMethod() 호출
ref3.rideMethod(); // Child.rideMethod() 호출
ref1.loadMethod(); // GrandParents.loadMethod() 호출
ref2.loadMethod(1); // Parents.loadMethod(int n) 호출
ref3.loadMethod(1.2); // Child.loadMethod(double n) 호출
}
}
GrandParents타입의 참조변수 ref1은 Child타입의 인스턴스를 참조한다. ref2와 ref3도 마찬가지로 Child인스턴스를 각각 참조한다. 즉, 참조변수의 타입은 모두 다르지만, 최하위 클래스인 Child의 인스턴스를 참조한다.
Child's rideMethod
Child's rideMethod
Child's rideMethod
GrandParants' loadMethod
Parants' loadMethod
Child's loadMethod
출력결과를 보면 참조변수의 자료형에 관계없이 최하위 클래스인 Child의 rideMethod만 호출되었다는 사실을 알 수 있다. 이에 비해 loadMethod는 각각 호출부분도 다르고 결과도 각각 다르다. 오버라이딩에 비해 오버로딩된 메소드는 사실상 이름만 같을 뿐 서로 다른 메소드라고 볼 수 있다.
- 상위 클래스의 참조변수는 하위 클래스의 인스턴스를 참조할 수 있다.
- 오버라이딩된 상위 클래스의 메소드는 오버라이딩을 한 하위 클래스의 메소드에 의해 가려진다. 즉, 외부에서는 참조변수를 통해서 오버라이딩된 상위 클래스의 메소드를 호출할 수 없다. (내부에서는 super키워드로 호출가능!)
참고블로그 :
'Java > Java' 카테고리의 다른 글
[Java] 상속 (0) | 2022.11.30 |
---|---|
[Java]public static void (0) | 2022.11.27 |
[Java] 클래스의 구성 (0) | 2022.11.22 |
[Java] 클래스 선언 (0) | 2022.11.22 |
Object-Oriented Programming(OOP) (0) | 2022.11.22 |
댓글