OOP의 개념(상속)
상속은 한 클래스를 확장하여 새로운 클래스를 만드는 것을 말한다. 이렇게 새로 만들어지는 클래스를 하위클래스(SubClass)라고 부른다. 그리고 원래의 클래스는 상위클래스(SuperClass)라고 부른다. 하위클래스는 상위클래스에서 정의한 메쏘드와 변수들을 그대로 가지고 있을 수 있다. (물론 이것도 접근변경자를 통해 조절이 가능하다) 여기에 추가로 하위 클래스 자체적으로 정의된 메쏘드와 변수들을 가질 수 있다. 어떤 클래스에 대해서, 상속받는 하위 클래스는 여러 개가 될수 있다.
이미 만들어놓은 클래스를 기반으로 원하는 기능이 추가된 새로운 클래스를 쉽게 만들어낼 수 있다
자바의 모든 클래스는 상속의 대상이 되는 상위클래스(SuperClass)가 반드시 하나 존재한다
프로그램의 재사용성과 확장성을 높인다.
class Employee {
String name; String id;
//생성자
public Employee(String name1, String id1) {
name = name1; id = id1;
}
public void gotoOffice() {
System.out.println(name+"님 출근하였습니다...");
}
public void gotoHome() {
System.out.println(name+"님 퇴근하였습니다...");
}
}
Employee Class를 상속한 Manager Class가 있다….
//직원클래스를 상속한 일반관리자 Class
class Manager extends Employee {
String chargeDept;
public Manager(String newName,String newID,String newDept) {
//super는 상위클래스의 생성자를 의미
super(newName, newID);
this.chargeDept = newDept;
}
public void startJob() {
System.out.println(this.chargeDept + " " + super.name + "님이 관리업무를 시작합니다...");
}
}
Employee Class를 상속한 SalesEmployee Class가 있다….
//직원클래스를 상속한 영업팀직원 클래스
class SalesEmployee extends Employee {
//영업담당지역, 메소드내에서만 변수에 접근이 가능하다.
private String chargeArea;
public SalesEmployee(String newName,String newID,String newArea) {
//super는 상위클래스의 생성자를 의미
super(newName, newID);
this.chargeArea = newArea;
}
public void startJob() {
System.out.println(super.name + "님이 “ + this.chargeArea + “ 지역으로 영업업무를 나갑니다...");
}
}
상속을 위해서는 extends keyword를 사용한다. (class a extends b b로 부터 상속을 받음)
extends키워드를 이용해 클래스를 상속받으면 상속받은 클래스는 상위클래스의 필드와 메소드를 상속받는다.
(예) 앞의 Manager Class instance화 했다면
Manager m = new Manager(“홍길동”, “12345”,”관리부”);
String id = m.id;
m.gotoOffice();
등과 같이 사용이 가능하다… 물론 m.startJob() 처럼 자신의 Method를 사용이 가능하다.
하위클래스는 객체를 생성할 때 상위클래스의 객체를 먼저 생성해야 한다. 하위클래스는 크게 상위클래스 객체부와 자기 자신의 객체부로 나눌 수 있다. 즉 new 키워드를 사용하여 하위클래스를 생성하면 상위클래스의 부분과 하위클래스 고유의 부분이 나뉘어 생성된다는 것이다. 이때 상위클래스 부분을 super라고 한다. 그리고 상위클래스 객체부를 생성하기 위해 사용하는 것이 바로 상위클래스 생성자 super()이다.
(예) 만약 Manager Class에서 super(newName, newID)를 생략하면 다음과 같은 에러가 난다. 생략시에는 default로 super() 이 삽입되는데ㅡ Employee에는 파라미터가 없는 생성자가 선언되지 않았기때문에 ….
Example.java:22: cannot resolve symbol symbol : constructor Employee ()
location: class Employee
public Manager(String newName,String newID,String newDept) {
하위클래스에서 상위클래스의 멤버를 호출하고자 할때는 .(점) 연산자를 사용하여 나타낼 수 있다.
(예) Manager Class에서 super.name
public void startJob() {
System.out.println(this.chargeDept +” “+ super.name + "님이 직원관리 업무 를 시작합니다...");
}
이제 SaleEmployee(영업부직원)을 상속한(확장한) SalesChief(영업팀장) Class를 만들어보자.
영업팀장 Class는 영업부직원과 같이 id,이름,영업담당 지역을 가지면서 자신만의 속성을 가진다.
//영업팀직원 클래스를 상속한 영업팀장 클래스
class SalesChief extends SalesEmployee {
int salesTarget; //영업팀 목표 매출액
public SalesChief(String newName,String newID,String newArea, int newSalesTarget) {
//super는 상위클래스의 생성자를 의미
super(newName, newID, newArea);
this.salesTarget = newSalesTarget;
}
//영업팀장의 업무는 더이상 확장이 안된다는 의미, 상속할수가 없다는 의미
final public void startJob() {
System.out.println(super.name + "님이 영업팀 직원을 관리한다...");
System.out.println(super.name + "님이 관리하는 영업팀의 매출목표는 "+ this.salesTarget + "만원 입니다...");
}
}
SalesChief Class의 경우 SalesEmployee의 startJob() 메소드를 재정의(Override) 했다.
메소드의 재정의라고 하는 것은 상속의 관계에 있는 클래스들 사이에서 상위클래스에 있는 메소드를 하위클래스에서 다시 정의하여 사용하는 것을 말한다. 이 경우 하위클래스의 메소드는 상위클래스의 메소드와 이름도, 매개변수 타입도, 그리고 반환형도 같다. 하지만 내부적으로 하는 일은 다르다.
만약 SalesChief Class의 startJob 메소드에서 다음과 같이 정의한다면,,,
public void startJob() {
System.out.println(super.chargeArea);
System.out.println(super.name + "님이 영업팀 직원을 관리한다...");
System.out.println(super.name + "님이 관리하는 영업팀의 매출목표는 "+ this.salesTarget + "만원 입니다...");
}
아래와 같은 오류발생…
Example.java:59: chargeArea has private access in SalesEmployee
System.out.println(super.chargeArea);
객체지향 언어에서는 정보의 은닉(캡슐화,Encapsulation) 을 위해서 위와 같은 접근지정을 할 수 있다. 즉 SaleEmployee의 자료중 정보 은닉의 필요가 있는 필드나 메소드는 접근지정자를 사용해 접근을 제한할 수 있는 것이다.
현재 작성한 클래스나 클래스의 멤버를 다른 클래스에게 상속시키고자 하지 않을경우엔 final 키워드를 사용한다.예를 들어 한 회사에 영업팀장은 그 만의 고유한 업무를 가지고 영업팀장의 업무를 관리하는 또 다른 개체가 없을 때는 영업팀장을 상속하여 더 확장된 기능을 가진 클래스를 만들 필요가 없을 것이다. 이럴 경우 final 키워드를 사용한다.
메소드의 경우도 이와 같이 상속을 완전히 금지하는 경우가 있는데 이러한 메소드도 final이라는 키워드와 같이 사용하게 되고 종단 메소드(final method)라고 부른다.
-------------------------------
/* Class의 상속에 관한 예제 Example1.*/
//직원 Class
class Employee {
String name;
String id;
//생성자
public Employee(String name1, String id1) {
name = name1;
id = id1;
}
public void gotoOffice() {
System.out.println(name+"님 출근하였습니다...");
}
public void gotoHome() {
System.out.println(name+"님 퇴근하였습니다...");
}
}
//직원클래스를 상속한 일반관리자 Class
class Manager extends Employee {
String chargeDept;
public Manager(String newName,String newID,String newDept) {
//super는 상위클래스의 생성자를 의미
super(newName, newID); this.chargeDept = newDept;
}
public void startJob() {
System.out.println(this.chargeDept + " " + super.name + "님이 일을 시작합니다...");
}
}
//직원클래스를 상속한 영업팀직원 클래스
class SalesEmployee extends Employee {
//영업담당지역, 메소드내에서만 변수에 접근이 가능하다.
private String chargeArea;
public SalesEmployee(String newName,String newID,String newArea) {
//super는 상위클래스의 생성자를 의미
super(newName, newID); this.chargeArea = newArea;
}
public void startJob() {
System.out.println(super.name + "님이 " + this.chargeArea + " 지역으로 영업업무를 나갑니다...");
}
}
//영업팀직원 클래스를 상속한 영업팀장 클래스
class SalesChief extends SalesEmployee {
int salesTarget; //영업팀 목표 매출액
public SalesChief(String newName,String newID,String newArea, int newSalesTarget) {
super(newName, newID, newArea); this.salesTarget = newSalesTarget;
}
public void startJob() {
System.out.println(super.name + "님이 영업팀 직원을 관리한다...");
System.out.println(super.name + "님이 관리하는 영업팀의 매출목표는 "+ this.salesTarget + "만원 입니다...");
}
}
//Main Class
class Example {
public static void main(String args[]) {
Manager m = new Manager("이종철","12345","솔루션개발");
m.gotoOffice(); m.startJob(); m.gotoHome();
SalesEmployee se = new SalesEmployee("차두리","23456","서울");
se.gotoOffice(); se.startJob(); se.gotoHome();
SalesChief sc = new SalesChief("홍길동","34567","전국",9000);
sc.gotoOffice(); sc.startJob(); sc.gotoHome();
}
}