상속은 상위 클래스의 기능을 하위 클래스에게 물려주는 기능이다. 그렇다면 하위 클래스는 상위 클래스의 메소드를 주어진 그대로 사용해야 할까? 만약 그래야 한다면 제약이 상당할 것이다. 이런 제약을 벗어나려면 하위 클래스가 부모 클래스의 기본적인 동작 방법을 변경할 수 있어야 한다. 이런 맥락에서 도입된 기능이 메소드 오버라이딩(overriding)이다.
아래 예제는 클래스 Calculator의 기본적인 동작 방법을 상속 받은 SubCalculator에 빼기 기능을 추가하고 있다. 이것은 상위 클래스의 기능에 새로운 기능을 추가한 것이다. 만약 상위 클래스에서 물려 받은 sum()을 호출했을 때 아래와 같이 그 결과를 좀 더 친절하게 알려줘야 한다면 어떻게 해야할까?
> 실행 결과는 30입니다.
Calculator의 예제를 변경해보자.
class Calculator {
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);
}
public void avg() {
System.out.println((this.left + this.right) / 2);
}
}
class SubCalculator extends Calculator {
/* 추가 */
public void sum() {
System.out.println("실행 결과는 " +(this.left + this.right)+"입니다.");
}
public void substract() {
System.out.println(this.left - this.right);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
SubCalculator c1 = new SubCalculator();
c1.setNum(10, 20);
c1.sum();
c1.avg();
c1.substract();
}
}
/* 실행 결과 */
> 실행 결과는 30입니다.
> 15
> -10
sum()이 SubCalculator에 추가되었다. 실행 결과는 c1.sum()이 상위 클래스의 메소드가 아니라 하위 클래스의 메소드 sum()을 실행하고 있다는 것을 보여준다. 하위 클래스 입장에서 부모 클래스란 말하자면 기본적인 동작 방법을 정의한 것이라고 생각할 수 있다. 하위 클래스에서 상위 클래스와 동일한 메소드를 정의하면 부모 클래스로부터 물려 받은 기본 동작 방법을 변경하는 효과를 갖게 된다. 기본동작은 폭넓게 적용되고, 예외적인 동작은 더 높은 우선순위를 갖게하고 있다. 이것은 공학에서 일반적으로 발견되는 규칙이다. 이것을 메소드 오버라이딩(overriding)이라고 한다.
상위 클래스에서 정의하고 있는 메소드 avg()는 계산 결과를 화면에 출력하고 있다. 그런데 계산 결과를 좀 더 다양하게 사용하기 위해서 메소드 avg()가 화면에 결과를 출력하는 대신 계산 결과를 리턴해주면 좋겠다. 그래서 아래와 같이 코드를 고쳐봤다.
class Calculator {
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);
}
public void avg() {
System.out.println((this.left + this.right) / 2);
}
}
class SubCalculator extends Calculator {
public void sum() {
System.out.println("실행 결과는 " +(this.left + this.right)+"입니다.");
}
public int avg() {
return (this.left + this.right) / 2;
}
public void substract() {
System.out.println(this.left - this.right);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
SubCalculator c1 = new SubCalculator();
c1.setNum(10, 20);
c1.sum();
c1.avg();
c1.substract();
}
}
이것은 에러를 발생한다. 오버라이딩(overriding)을 하기 위해서는 메소드의 리턴 형식이 같아야 한다. 즉 클래스 Calculator의 메소드 avg()는 리턴 타입이 void이다. 그런데 이것을 상속한 클래스 SubCalculator의 리턴 타입은 int이다. 오버라이딩을 하기 위해서는 아래의 조건을 충족시켜야 한다.
- 메소드의 이름
- 메소드 매개변수의 숫자와 데이터 타입 그리고 순서
- 메소드의 리턴 타입
위와 같이 메소드의 형태를 정의하는 사항들을 통틀어 메소드의 서명(signature)이라고 한다. 즉, 위의 에러는 메소드들 간의 서명이 달라서 발생한 문제다. 아래와 같이 상위 클래스의 코드를 변경해서 이 문제를 우회하자.
class Calculator {
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);
}
/* 변경 */
public int avg() {
return (this.left + this.right) / 2;
}
}
class SubCalculator extends Calculator {
public void sum() {
System.out.println("실행 결과는 " +(this.left + this.right)+"입니다.");
}
public int avg() {
return (this.left + this.right)/2;
}
public void substract() {
System.out.println(this.left - this.right);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
SubCalculator c1 = new SubCalculator();
c1.setNum(10, 20);
c1.sum();
c1.avg();
c1.substract();
}
}
상위 클래스와 하위 클래스의 서명이 같기 때문에 메소드 오버라이딩을 할 수 있었다. 그런데 위의 코드를 보면 중복이 발생했다. 메소드 avg()의 부모와 자식 클래스가 같은 로직을 가지고 있다. 중복은 제거되어야 한다. 생성자와 마찬가지로 super를 사용하면 이 문제를 해결할 수 있다.
class Calculator {
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);
}
public int avg() {
return ((this.left + this.right) / 2);
}
}
class SubCalculator extends Calculator {
public void sum() {
System.out.println("실행 결과는 " +(this.left + this.right)+"입니다.");
}
public int avg() {
return super.avg(); // 변경
}
public void substract() {
System.out.println(this.left - this.right);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
SubCalculator c1 = new SubCalculator();
c1.setNum(10, 20);
c1.sum();
System.out.println("실행 결과는" + c1.avg());
c1.substract();
}
}
하위 클래스의 메소드 avg()에서 상위 클래스의 메소드를 호출하기 위해서 super를 사용하고 있다. 덕분에 코드의 중복을 제거할 수 있었다.
이렇게 부모 클래스의 기능을 변경할 수 있는 방법인 메소드 오버라이딩에 대해서 알아봤다.