하위 클래스가 생성되는 과정
- 하위 클래스를 생성하면 상위 클래스가 먼저 생성됨
- new VIPCustomer()를 호출하면 Customer() 가 먼저 호출됨 (상위 클래스가 먼저 호출됨)
- 클래스가 상속 받은 경우 하위 클래스의 생성자에서는 반드시 상위 클래스의 생성자를 호출함
Customer.java
public Customer() { // 생성자 (일반 고객)
customerGrade = "SILVER";
bonusRatio = 0.01;
System.out.println("Customer() 생성자 호출"); // 생성자 안에 작성
}
VIPCustomer.java
public VIPCustomer() {
customerGrade = "VIP"; // 오류 발생 상속받은 Customer에서 customerGrade가 private이기 때문
bonusRatio = 0.05;
salesRatio = 0.1;
System.out.println("VIPCustomer() 생성자 호출"); // 생성자 안에 작성
}
super 키워드
- 하위 클래스에서 가지는 상위 클래스에 대한 참조값
- super()는 상위 클래스의 기본 생성자를 호출함
- 하위 클래스에서 명시적으로 상위 클래스의 생성자를 호출하지 않으면 super()가 호출됨 (이때 반드시 상위 클래스의 기본 생성자가 존재 해야 함)
- 상위 클래스의 기본 생성자가 없는 경우 (다른 생성자가 있는 경우) 하위 클래스에서는 생성자에서는 super를 이용하여 명시적으로 상위 클래스의 생성자를 호출함
- super는 생성된 상위 클래스 인스턴스의 참조값을 가지므로 super를 이용하여 상위 클래스의 메서드나 멤버 변수에 접근할 수 있음
매개변수를 달리할 경우 생성자를 여러개 만들 수 있다.
하위 클래스에서 상위 클래스의 생성자를 명시적으로 호출해야 할 때는 상위 클래스의 생성자에 맞춰서 호출해줘야한다. (상위 클래스의 생성자가 매개변수가 있으면, 매개변수를 넣어서 호출해주거나...)
Customer.java
// 매개변수가 있는 생성자의 경우
public Customer(int customerID, String customerName) {
this.customerID = customerID; // this는 인스턴스가 자기자신의 주소를 가지고 있을 때,
this.customerName = customerName; // class가 생성된 자기자신의 주소를 가지고 있는 값 = this
// 추가적으로 this는 클래스의 또다른 생성자를 호출할 수 있는 기능이 있다.
customerGrade = "SILVER";
bonusRatio = 0.01;
System.out.println("Customer(int, String) 생성자 호출");
}
VIPCustomer.java
// super를 이용하여 상위 클래스의 생성자 명시적으로 호출
public VIPCustomer(int customerID, String customerName) {
super(customerID, customerName); // super는 하위 클래스가 상위 클래스이 참조값을 가지게 된다.
// super.하면 메소드도 다 볼 수 있고, 호출할 수도 있음(super.안쓰고 그냥 메소드 이름만 써도 가능)
// super에서도 생성자를 호출할 수 있는 기능이 있음
// 만약 super()만 쓰면 default 생성자 호출함
// super()를 명시하지 않으면 compiler가 dafault constructor를 넣어준다.
customerGrade = "VIP";
bonusRatio = 0.05;
salesRatio = 0.1;
System.out.println("VIPCustomer(int, String) 생성자 호출");
}
CustomerTest.java
package ch03;
public class CustomerTest {
public static void main(String[] args) {
//Customer customerLee = new Customer();
//customerLee.setCustomerName("이순신");
//customerLee.setCustomerID(10010);
Customer customerLee = new Customer(10010, "이순신");
customerLee.bonusPoint = 1000;
System.out.println(customerLee.showCustomerInfo());
//VIPCustomer customerKim = new VIPCustomer();
//customerKim.setCustomerName("김유신");
//customerKim.setCustomerID(10020);
VIPCustomer customerKim = new VIPCustomer(10020, "김유신");
customerKim.bonusPoint = 10000;
System.out.println(customerKim.showCustomerInfo());
}
}
super()를 사용한 경우
public VIPCustomer(int customerID, String customerName) {
//super(customerID, customerName);
super();
customerGrade = "VIP";
bonusRatio = 0.05;
salesRatio = 0.1;
System.out.println("VIPCustomer(int, String) 생성자 호출");
}
상속에서 인스턴스 메모리의 상태
- 항상 상위 클래스의 인스턴스가 먼저 생성되고, 하위 클래스의 인스턴스가 생성됨 (그림 순서상으로는 1 생성 후 2 생성)
- private도 다 만들어진다. 단지 접근만 못하는 것임
형 변환(업캐스팅)
1. 상위 클래스로 변수를 선언하고 하위 클래스의 생성자로 인스턴스를 생성
// 사용 예시
Customer customerLee = new VIPCustomer();
// 이렇게 사용할 수 있는 이유는 VIPCustomer가 Customer type을 내포하고 있기 때문
2. 상위 클래스 타입의 변수에 하위 클래스 변수가 대입
VIPCustomer vCustomer = new VIPCustomer();
addCustomer(vCustomer);
// addCustomer의 매개변수를 보면 Customer type으로 받아준다.
// 이렇게 매개변수로 넘겨줘도 받는다.
// VIPCustomer type이어도 Customer type을 내포하고 있기 때문
int addCustomer(Customer customer) {
}
하위 클래스는 상위 클래스의 타입을 내포하고 있으므로 상위 클래스의 묵시적 형 변환이 가능함
상속 관계에서 모든 하위 클래스는 상위 클래스로 형 변환(업캐스팅)이 됨 (그 역은 성립하지 않음)
형 변환과 메모리
- Customer vc = new VIPCustomer(); 에서 vc가 가리키는 것은? 그림의 1번 부분
- VIPCustomer() 생성자에 의해 VIPCustomer 클래스의 모든 멤버 변수에 대한 메모리는 생성되었지만, 변수의 타입이 Customer이므로 실제 접근 가능한 변수나 메서드는 Customer의 변수와 메서드임
클래스의 계층구조가 여러 단계인 경우
Human은 내부적으로 Promate와 mammal의 타입을 모두 내포하고 있음
Primate pHumman = new Humman();
Mammal mHumman = new Humman();
2023 KAKAO Tech Campus_BackEnd 필수 과정
Java 3주차 강의 정리 내용입니다.
'개발 > Java' 카테고리의 다른 글
[3주차] Java 중급 - 메서드 재정의와 가상 메서드 원리 (0) | 2023.06.25 |
---|---|
[3주차 Java 중급] - 메서드 재정의 하기(overriding) (0) | 2023.06.25 |
[3주차] Java 중급 - 상속을 활용한 멤버십 클래스 구현하기 (0) | 2023.06.24 |
[3주차] Java 중급 - 객체 간의 상속은 어떤 의미일까 (0) | 2023.06.24 |
[1주차] Java 중급 - 복습해봅시다(static과 싱글톤 패턴) (0) | 2023.06.24 |