우리의 계산기는 2개의 값(left, right)에 대한 연산(sum, avg) 만을 수행 할 수 있다. 그런데 만약 3개의 값을 대상으로 연산을 해야 한다면 어떻게 해야할까?
우선 아래와 같이 입력값을 3개 받아야 할 것이다.
c1.setOprands(10, 20, 30);
이를 위해서 기존의 setOprands 메서드를 아래와 같은 모습을 수정한다면 2개의 입력값을 받을 수 없게 될 것이다.
public void setOprands(int left, int right, int third){ this.left = left; this.right = right; this.third = third; }
이런 경우 아래와 같이 메서드의 이름을 변경하면 된다.
c1.setOprands2(10, 20); c1.setOprands3(10, 20, 30);
이것도 좋은 방법이지만 매개변수의 수에 따라서 메서드의 이름이 달라지는 것은 왠지 깔끔한 방법이 아닌 것 같다.
그럼 어떻게 해야 좋을까? 코드를 보자.
class Calculator{
int left, right;
int third = 0;
public void setOprands(int left, int right){
System.out.println("setOprands(int left, int right)");
this.left = left;
this.right = right;
}
// 메서드 오버로딩
public void setOprands(int left, int right, int third){
System.out.println("setOprands(int left, int right, int third)");
this.left = left;
this.right = right;
this.third = third;
}
public void sum(){
System.out.println(this.left+this.right+this.third);
}
public void avg(){
System.out.println((this.left+this.right+this.third)/3);
}
}
public class CalculatorDemo {
public static void main(String[] args) {
Calculator c1 = new Calculator();
c1.setOprands(10, 20);
c1.sum();
c1.avg();
c1.setOprands(10, 20, 30);
c1.sum();
c1.avg();
}
}
[실행결과]
setOprands(int left, int right)
30
15
setOprands(int left, int right, int third)
60
30
✔️ 메서드 오버로딩(overloading)
: 이름은 같지만 시그니처는 다른 메소드를 중복으로 선언 할 수 있는 방법
: 매개변수의 숫자에 따라서 같은 이름의, 서로 다른 메서드를 호출
결론적으로 말하면 메서드 오버로딩은 매개변수를 사용한다.
즉, 매개변수가 다르면 이름이 같아도 서로 다른 메서드가 되는 것이다.
반면에 매개변수는 같지만 리턴타입이 다르면 오류가 발생한다. 아래의 코드를 보자.
public class OverloadingDemo {
void A () {
System.out.println("void A()");
}
void A (int arg1) {
System.out.println("void A (int arg1)");
}
void A (String arg1) {
System.out.println("void A (String arg1)");
}
// Error
int A (){
System.out.println("void A()");
}
public static void main(String[] args) {
OverloadingDemo od = new OverloadingDemo();
od.A();
od.A(1);
od.A("coding everybody");
}
}
1번째와 2번째 메서드 A는 매개변수의 숫자가 다르다.
2번째 3번째 메서드 A는 인자의 숫자는 같지만 매개변수의 데이터 타입이 다르다.
이런 경우는 오버로딩이 가능하다.
메서드를 호출 할 때 전달되는 인자의 데이터 타입에 따라서 어떤 메소드를 호출할지를 자바가 판단 할 수 있기 때문이다.
하지만 메서드의 리턴값은 메소드를 호출하는 시점에서 전달되지 않는 정보이기 때문에 오버로딩의 대상이 될 수 없다.
상속의 관계에서도 오버로딩을 사용할 수 있을까? 물론이다.
위의 코드를 상속받아 적어보았다.
public class OverloadingDemo2 extends OverloadingDemo {
void A (String arg1, String arg2) {
System.out.println("sub class : void A (String arg1, String arg2)");
}
@Override
void A () {
System.out.println("sub class : void A ()");
}
public static void main(String[] args) {
OverloadingDemo2 od = new OverloadingDemo2();
od.A();
od.A(1);
od.A("coding everybody");
od.A("coding everybody", "coding everybody");
}
}
[결과값]
sub class : void A ()
void A (int arg1)
void A (String arg1)
sub class : void A (String arg1, String arg2)
클래스 OverloadingDemo2는 OverloadingDemo을 상속 받고 있다.
OverloadingDemo2의 첫번째 메서드 A는 문자열을 데이터타입으로 하는 두개의 매개변수를 가지고 있다. 이러한 형태의 변수는 부모 클래스에서는 정의되어 있지 않기 때문에 메서드 오버로딩이 되는 것이다.
반면에 두번째 메서드 A는 매개변수가 없다.
그리고 부모 클래스에는 이미 매개변수가 없는 메서드 A가 존재한다.
이 둘은 매개변수의 형태가 같기 때문에 오버로딩이 아니라 오버라이딩에 해당한다.
✔️ overloading (올라탄다 / 덮는다)
: 부모 클래스의 메서드의 동작방법을 변경.
: 물리적으로 완전히 다른 메서드 정의 (중복정의)
: 한 클래스 내부
✔️ overriding
: 같은 이름, 다른 매개변수의 메서드들을 여러개 만듬
: 물리적으로 완전히 같은 메서드 정의 (재정의)
: 상속을 전제