Chapter 12. 상속 다루기
✏️ 메서드 올리기
배경
- 메서드 올리기를 적용하기 가장 쉬운 상황은 메서드들의 본문 코드가 똑같을 때다.
- 이럴 땐 그냥 복사 - 붙여넣기면 끝이다.
- 보통 메서드 올리기를 적용하려면 선행 단계가 필요하다.
- e.g. 서로 다른 두 클래스의 두 메서드를 같은 메서드로 만들기
- 각각의 함수를 매개변수화한 후 메서드를 상속 계층의 위로 올리기
- e.g. 서로 다른 두 클래스의 두 메서드를 같은 메서드로 만들기
절차
- 같은 동작의 메서드인지 확인한다.
- 메서드 안에서 호출하는 다른 메서드와 참조하는 필드들을 슈퍼클래스에서도 호출하고 참조할 수 있는지 확인한다.
- 함수 선언 바꾸기로 슈퍼클래스에서 사용하고 싶은 형태로 통일한다.
- 서브클래스 중 하나의 메서드를 제거한다.
- 모든 서브클래스의 메서드가 없어질 때까지 하나씩 제거한다.
예시
Before
// 코드 내용: Party 클래스를 상속한 Employee, Department 클래스에서 동일한 메서드 발견
// 슈퍼클래스에는 없지만 서브 클래스에 존재하는 상태
class Employee extends Party {
get annualCost() {
return this.monthlyCost * 12;
}
}
class Department extends Party {
get totalAnnualCost() {
return this.monthlyCost * 12;
}
}
After
- 함수 선언 바꾸기로 이름 통일 (annualCost)
- 슈퍼클래스인 Party로 annualCost 메서드 올리기
class Party {
get annualCost() {
return this.monthlyCost * 12;
}
}
class Department extends Party {
}
class Employee extends Party {
}
✏️ 필드 올리기
배경
- 서브클래스들이 독립적으로 개발되었거나 뒤늦게 하나의 계층구조로 리팩토링된 경우 일부 기능이 중복될 수 있다.
- 특히 필드
- 필드들이 비슷한 방식으로 쓰인다고 판단되면 슈퍼클래스로 끌어올린다.
- 이렇게 하면 두 가지 중복을 줄일 수 있다.
- 데이터 중복 선언 없애기
- 해당 필드를 사용하는 동작을 슈퍼클래스로 옮기기 가능
절차
- 후보 필드들이 사용하는 곳에서 동일한 방식으로 사용하는지 확인한다.
- 필드 이름 바꾸기로 같은 이름으로 변경한다.
- 슈퍼클래스에 새로운 필드를 생성하고 서브클래스 필드들을 제거한다.
예시
Before
// 코드 내용 : 서브클래스에서 동일하게 사용되는 name 필드 발견
class Employee {}
class Salesperson extends Employee {
#name;
}
class Engineer extends Employee {
#name;
}
After
- 슈퍼클래스로 name 필드를 옮기고 서브클래스에서는 제거
class Employee {
#name;
}
class Salesperson extends Employee {
}
class Engineer extends Employee {
}
✏️ 생성자 본문 올리기
배경
- 이번 리팩토링이 간단히 끝날 것 같지 않다면 생성자를 팩토리 함수 바꾸기를 고려해본다.
절차
- 슈퍼클래스에 생성자가 없다면 하나 정의한다.
- 문장 슬라이드하기로 공통 문장 모두를 super() 호출 직후로 옮긴다.
- 공통 코드를 슈퍼클래스에 추가하고 서브클래스들을 제거한다.
- 생성자 매개변수 중 공통 코드에서 참조하는 값들을 모두 super()로 건넨다.
- 생성자 시작 부분으로 옮길 수 없는 공통 코드에는 함수 추출하기와 메서드 올리기를 차례로 적용한다.
예시
Before
// 코드 내용: 서브클래스 생성자에 공통으로 사용되는 name 발견
class Party {}
class Employee extends Party {
constructor(name, id, monthlyCost) {
super();
this._id = id;
this._name = name; 📍
this._monthlyCost = monthlyCost;
}
}
class Department extends Party {
constructor(name, staff) {
super();
this._name = name; 📍
this._staff = staff;
}
}
After
- Employee의 name을 super 아래로 옮긴다. (문장 슬라이드하기)
- Party 클래스에 name을 추가하고 이 인수를 슈퍼클래스 생성자에 매개변수로 건넨다.
class Party {
constructor(name) {
this._name = name;
}
}
class Employee extends Party {
constructor(name, id, monthlyCost) {
super(name);
this._id = id;
this._monthlyCost = monthlyCost;
}
}
class Department extends Party {
constructor(name, staff) {
super(name);
this._staff = staff;
}
}
✏️ 메서드 내리기
배경
- 특정 서브클래스 하나(혹은 소수)만 관련된 메서드는 슈퍼클래스에서 제거하고 해당 서브클래스에 추가하는 편이 깔끔하다.
절차
- 대상 메서드를 모든 서브클래스에 복사한다.
- 슈퍼클래스에서 그 메서드를 제거한다.
- 이 메서드를 사용하지 않는 모든 서브클래스에서 제거한다.
예시
Before
// 코드 내용: 슈퍼클래스에 있는 quota 메서드는 Salesperson 클래스에서만 사용중
class Employee {
get quota() {}
}
class Engineer extends Employee {}
class Salesperson extends Employee {}
After
- Salesperson 서브클래스로 메서드를 옮긴 후 슈퍼클래스와 나머지 서브 클래스들에서 제거
class Employee {
}
class Engineer extends Employee {}
class Salesperson extends Employee {
get quota() {}
}
✏️ 필드 내리기
배경
- 서브클래스 하나(혹은 소수)에서만 사용하는 필드는 해당 서브클래스(들)로 옮긴다.
절차
- 대상 필드를 모든 서브클래스에 정의한다.
- 슈퍼클래스에서 그 필드를 제거한다.
- 이 필드를 사용하지 않는 모든 서브클래스에서 제거한다.
예시
Before
// 코드 내용: 슈퍼클래스에 있는 quota 필드는 Salesperson에서만 사용중
class Employee {
#quota;
}
class Engineer extends Employee {}
class Salesperson extends Employee {}
After
- Salesperson 서브클래스로 필드를 옮긴 후 슈퍼클래스와 나머지 서브 클래스들에서 제거
class Employee {
}
class Engineer extends Employee {}
class Salesperson extends Employee {
#quota;
}
✏️ 타입 코드를 서브클래스로 바꾸기
배경
- 소프트웨어 시스템에서는 비슷한 대상들을 특정 특성에 따라 구분해야 할 때가 있다.
- 직원을 담당 업무로 구분
- 주문을 시급성으로 구분
- 이런 일을 다루는 수단으로는 타입 코드(type code) 필드가 있다.
- 열거형, 심볼, 문자열, 숫자 등
- 타입 코드 이상으로 무언가가 필요할 때 서브클래스를 사용하면 좋다.
- 조건에 따라 다르게 동작하도록 해주는 다형성 제공
- 특정 타입에서만 의미 있는 값을 사용하는 필드나 메서드가 있을 때 활용성 증대
절차
- 타입 코드 필드를 자가 캡슐화한다.
- 타입 코드 값 하나를 선택해 서브클래스를 만든다.
- 매개변수로 받은 타입 코드와 서브클래스를 매핑하는 선택 로직을 만든다.
- 타입 코드 값 각각에 대해 서브클래스 생성과 선택 로직 추가를 반복한다.
- 타입 코드를 제거한다.
- 타입 코드 접근자를 이용하는 메서드 모두에 메서드 내리기와 조건부 로직을 다형성으로 바꾸기를 적용한다.
예시
Before
// 코드 내용 : engineer, manager, salesperson 타입을 체크하는 validateType 함수
class Employee {
constructor(name, type) {
this.validateType(type);
this._name = name;
this._type = type;
}
validateType(arg) {
if (!['engineer', 'manager', 'salesperson'].includes(arg)) {
throw new Error(`${arg}라는 직원 유형은 없습니다.`);
}
}
toString() {
return `${this._name} (${this._type})`;
}
}
After
- 타입별로 서브클래스 생성
- createEmployee 팩토리 함수로 선택 로직 구현
- 각 타입별로 서브클래스 생성자 리턴
class Employee {
constructor() {
this._name = name;
}
}
class Engineer extends Employee {
get type() {
return 'engineer';
}
}
class Salesperson extends Employee {
get type() {
return 'salesperson';
}
}
function createEmployee(name, type) {
switch(type) {
case 'engineer': return new Engineer();
case 'salesperson': return new Salesperson();
default: throw new Error(`${type}라는 직원 유형은 없습니다.`);
}
}
✏️ 서브클래스 제거하기
배경
- 서브클래싱은 원래 데이터 구조와는 다른 변종을 만들거나 종류에 따라 동작이 달라지게 할 수 있는 유용한 메커니즘이다.
- 다만 더 이상 쓰이지 않는 서브클래스는 시간 낭비를 유발할 뿐이므로 제거하는 것이 좋다.
절차
- 서브클래스의 생성자를 팩토리 함수로 바꾼다.
- 서브클래스의 타입을 검사하는 코드가 있다면 그 코드를 함수 추출하기와 함수 옮기기를 차례로 적용해 슈퍼클래스로 옮긴다.
- 서브클래스의 타입을 나타내는 필드를 슈퍼클래스에 만든다.
- 서브클래스를 참조하는 메서드가 방금 만든 타입 필드를 이용하도록 수정한다.
- 서브클래스를 제거한다.
예시
Before
// 코드 내용: 단순한 역할을 하는 서브 클래스들
class Person {
constructor(name) {
this._name = name;
}
get name() { return this._name; }
get genderCode() { return 'X'; }
}
class Male extends Person {
get genderCode() { return 'M'; }
}
class Female extends Person {
get genderCode() { return 'F'; }
}
// 클라이언트
const numberOfMales = people.filter((p) => p instanceof Male).length;
After
// 직관적인 방법 : 팩토리 함수
function createPerson(name) {
return new Person(name);
}
function createMale(name) {
return new Male(name);
}
function createFemale(name) {
return new Female(name);
}
// 내부 메서드 생성 방법 📍
// Person 클래스에 createPerson 함수를 제작해 기존 서브클래스가 하던 일을 대체
class Person {
constructor(name, genderCode) {
this._name = name;
this._genderCode = genderCode;
}
get name() { return this._name; }
get genderCode() { return this._genderCode; }
function createPerson(record){
switch(record.gender){
case 'M':
return new Person(record.name, 'M')
case 'F':
return new Person(record.name, 'F')
default:
return new Person(record.name, 'X')
}
}
get isMale() { return this.#genderCode === 'M'; }
}
// 클라이언트
const numberOfMales = people.filter((person) => person.isMale).length;
✏️ 슈퍼클래스 추출하기
배경
- 비슷한 일을 수행하는 두 클래스가 보이면 상속 메커니즘을 이용해 비슷한 부분을 공통의 슈퍼클래스로 옮겨 담을 수 있다.
- 공통된 부분이 데이터라면
- 필드 올리기
- 동작이라면
- 메서드 올리기
- 공통된 부분이 데이터라면
- 슈퍼클래스 추출하기의 대안으로는 클래스 추출하기가 있다.
- 어느 것을 선택하느냐는 중복 동작을 상속으로 해결하느냐 위임으로 해결하느냐에 달렸다.
절차
- 빈 슈퍼클래스를 만든다. 원래의 클래스들이 새 클래스를 상속하도록 한다.
- 생성자 본문 올리기, 메서드 올리기, 필드 올리기를 차례로 적용해 공통 원소를 슈퍼클래스로 옮긴다.
- 서브클래스에 남은 메서드들을 검토한다. 공통된 부분은 함수로 추출한 다음 메서드 올리기를 적용한다.
예시
Before
// 코드 내용 : 비용, 이름 등 공통된 기능을 수행하는 듯한 두 클래스
class Employee {
constructor(name, id, monthlyCost) {
this._id = id;
this._name = name;
this._monthlyCost = monthlyCost;
}
get monthlyCost() { return this._monthlyCost; } 월간 비용
get name() { return this._name; } 이름
get id() { return this._id; } 아이디
get annualCost() { return this.monthlyCost * 12; } 연간 비용
}
class Department {
constructor(name, staff) {
this._name = name;
this._staff = staff;
}
get staff() { return this._staff.slice(); }
get name() { return this._name; }
get totalMonthlyCost() {
return this.staff
.map(e => e.monthlyCost)
.reduce((sum, cost) => sum + cost);
}
get headCount() { return this.staff.length; } 직원 수
get totalAnnualCost() { return this.totalMonthlyCost * 12; } 총 연간 비용
}
After
- Party라는 슈퍼클래스를 생성하고 공통된 메서드를 옮긴다. (메서드 올리기)
- 슈퍼클래스에서 상속받은 필드, 메서드를 활용해 각 서브클래스에서는 고유의 역할만을 수행한다.
class Party {
constructor(name) {
this._name = name;
}
get name() { return this._name; }
get annualCost() {
return this.monthlyCost * 12;
}
}
class Employee extends Party {
constructor(name, id, monthlyCost) {
super(name);
this._id = id;
this._monthlyCost = monthlyCost;
}
get monthlyCost() { return this._monthlyCost; }
get id() { return this._id; }
}
class Department extends Party {
constructor(name, staff) {
super(name);
this._staff = staff;
}
get staff() { return this._staff.slice(); }
get monthlyCost() { // 총 월간 비용
return this.staff
.map(e => e.monthlyCost)
.reduce((sum, cost) => sum + cost);
}
get headCount() {
return this.staff.length;
}
}
✏️ 계층 합치기
배경
- 어떤 클래스와 그 부모가 너무 비슷해져서 더는 독립적으로 존재해야 할 이유가 사라지는 경우
- 바로 그 둘을 합쳐야 할 시점이다.
절차
- 두 클래스 중 제거할 것을 고른다.
- 필드 올리기와 메서드 올리기, 혹은 필드 내리기와 메서드 내리기를 적용해 하나의 클래스로 옮긴다.
- 제거할 클래스를 참조하던 모든 코드가 남겨질 클래스를 참조하도록 고친다.
- 빈 클래스를 제거한다.
예시
Before
class Employee {...}
class Salesperson extends Employee {...}
After
class Employee {...}
✏️ 서브클래스를 위임으로 바꾸기
배경
- 상속에는 단점이 있다.
- 가장 명확한 담점은 한 번만 쓸 수 있는 카드라는 것이다.
- 또 다른 문제로, 상속은 클래스들의 관계가 아주 긴밀하게 결합한다.
- 부모를 수정하면 이미 존재하는 자식들의 기능을 해치기가 쉽다.
- 위임은 이 두 문제를 모두 해결해준다.
- 객체 사이의 일반적인 관계이므로 상호작용에 필요한 인터페이스를 명확히 정의할 수 있다.
- 즉, 상속보다 결합도가 훨씬 약하다.
- “(클래스) 상속보다는 (객체) 컴포지션을 사용하라!”
- 여기서 컴포지션은 사실상 위임과 같은 말이다.
- 처음에는 상속으로 접근한 다음, 문제가 생기기 시작하면 위임으로 갈아탄다 (저자의 방법)
절차
- 생성자를 호출하는 곳이 많다면 생성자를 팩토리 함수로 바꾼다.
- 위임으로 활용할 빈 클래스를 만든다. 이 클래스의 생성자는 서브클래스에 특화된 데이터를 전부 받아야 하며, 보통은 슈퍼클래스를 가리키는 역참조도 필요하다.
- 위임을 저장할 필드를 슈퍼클래스에 추가한다.
- 서브클래스 생성 코드를 수정해 위임 인스턴스를 생성하고 위임 필드에 대입해 초기화한다.
- 서브클래스의 메서드 중 위임 클래스로 이동할 것을 고른다.
- 함수 옮기기를 적용해 위임 클래스로 옮긴다.
- 서브클래스 외부에도 원래 메서드를 호출하는 코드가 있다면 서브클래스의 위임 코드를 슈퍼클래스로 옮긴다.
- 서브클래스의 모든 메서드가 옮겨질 때까지 5~8 과정을 반복한다.
- 서브클래스들의 생성자를 호출하는 코드를 찾아서 슈퍼클래스의 생성자를 사용하도록 수정한다.
- 서브클래스를 제거한다.
예시
Before
talkback: 반응, 응답
delegate: 위임하다
// 문제 : 여러 요구사항이 추가될수록, 슈퍼클래스 수정이 필요해질수록 관리가 어려워짐
// 공연 예약 슈퍼클래스
class Booking {
constructor(show, date) {
this._show = show;
this._date = date;
}
// 관객과의 대화 시간을 성수기가 아닐 때만 제공
get hasTalkback() {
return this._show.hasOwnProperty('talkback') && !this.isPeakDay;
}
// 성수기, 비수기에 따라 가격 결정
get basePrice() {
let result = this._show.price;
if (this.isPeakDay) {
result += Math.round(result * 0.15);
}
return result;
}
}
// 프리미엄 예약 서브클래스
class PremiumBooking extends Booking {
constructor(show, date, extras) {
super(show, date);
this._extras = extras;
}
get hasTalkback() {
return this._show.hasOwnProperty('talkback');
}
// 슈퍼클래스의 가격에 프리미엄 피를 더한 값
get basePrice() {
return Math.round(super.basePrice + this._extras.PremiumFee);
}
// 슈퍼클래스에는 없는 기능
// 성수기가 아닐 때 저녁을 제공하는지 여부
get hasDinner() {
return this._extras.hasOwnProperty('dinner') && !this.isPeakDay;
}
}
After
- createPremiumBooking, createBooking 팩토리 함수 생성
- Booking 클래스 내 메서드에 위임이 존재할 때, 존재하지 않을 때의 조건 추가
- 각각의 고유한 기능들은 분리
class Booking {
constructor(show, date) {
this._show = show;
this._date = date;
}
// 위임이 존재하면 위임을 사용하는 분배 로직 추가
get hasTalkback() {
return this._premiumDelegate
? this._premiumDelegate.hasTalkback
: this._show.hasOwnProperty('talkback') && !this.isPeakDay;
}
get privateBasePrice() {
let result = this._show.price;
if (this.isPeakDay) {
result += Math.round(result * 0.15);
return result;
}
}
get basePrice() {
let result = this._show.price;
if (this.isPeakDay) {
result += Math.round(result * 0.15);
}
return this._premiumDelegate
? this._premiumDelegate.extendBasePrice(result)
: this._privateBasePrice;
}
// 위임이 존재할 때만 결과 도출
get hasDinner() {
return this._premiumDelegate ? this._premiumDelegate.hasDinner : undefined;
}
bePremium(extras) {
this._premiumDelegate = new PremiumBookingDelegate(this, extras);
}
// Booking 슈퍼클래스 생성
static createBooking(show, date) {
return new Booking(show, date);
}
// Premium 생성 메서드 (bePremium 메서드로 PremiumBookingDelegate 속성들 추가)
static createPremiumBooking(show, date, extras) {
const result = new Booking(show, date, extras);
result._bePremium(extras);
return result;
}
}
class PremiumBookingDelegate {
constructor(hostBooking, extras) {
this._host = hostBooking;
this._extras = extras;
}
get hasTalkback() {
return this._host._show.hasOwnProperty('talkback');
}
get basePrice() {
return Math.random(this._host._privateBasePrice + this._extras.PremiumFee);
}
get hasDinner() {
return this._extras.hasOwnProperty('dinner') && !this._host.isPeakDay;
}
extendBasePrice(base) {
return Math.round(base + this._extras.premiumFee);
}
}
✏️ 슈퍼클래스를 위임으로 바꾸기
배경
- 제대로 된 상속이라면 서브클래스가 슈퍼클래스의 모든 기능을 사용함은 물론, 서브클래스의 인스턴스를 슈퍼클래스의 인스턴스로도 취급할 수 있어야 한다.
- 슈퍼클래스가 사용되는 모든 곳에서 서브클래스의 인스턴스를 대신 사용해도 이상없이 동작해야 한다.
- 위임을 이용하면 기능 일부를 빌려올 뿐인, 서로 별개인 개념임이 명확해진다.
- 서브클래스 방식 모델링이 합리적일 때라도 슈퍼클래스를 위임으로 바꾸기도 한다.
- 서로 강하게 결합된 관계라서 슈퍼클래스를 수정하면 서브클래스가 망가지기 쉽기 때문이다.
- 위임에도 단점이 있다.
- 위임의 기능을 이용할 호스트의 함수 모두를 전달 함수로 만들어야 한다.
- 그럼에도 상속이 더는 최선의 방법이 아니게 되면 언제든 이번 리팩토링을 이용해 슈퍼클래스를 위임으로 바꿀 수 있다.
- 그래서 (웬만하면) 상속을 먼저 적용하고 (만일) 나중에 문제가 생기면 슈퍼클래스를 위임으로 바꾸는 것이 좋다.
절차
- 슈퍼클래스 객체를 참조하는 필드를 서브클래스로 만든다.
- 슈퍼클래스의 동작 각각에 대응하는 전달 함수를 서브클래스에 만든다.
- 슈퍼클래스의 동작 모두가 전달 함수로 오버라이드되었다면 상속 관계를 끊는다.
예시
Before
threshold: 한계점
revered : 존경받는
chrono : 시간, 시대와 관련됨을 나타냄
class CatalogItem {
constructor(id, title, tags) {
this._id = id;
this._title = title;
this._tags = tags;
}
get id() { return this._id; }
get title() { return this._title; }
hasTag(arg) { return this._tags.includes(arg);}
}
// 각 스크롤에는 ID, 제목이 있고 그 외 여러 태그가 붙어 있다.
// 스크롤에는 정기 세척 이력이 필요하기에 CatalogItem을 확장해 세척 관련 데이터 추가
// 사본이 여러 개지만 카탈로그 아이템은 하나일 수 있는 문제 발생
class Scroll extends CatalogItem {
constructor(id, title, tags, dataLastCleaned) {
super(id, title, tags);
this._lastCleaned = dataLastCleaned;
}
needsCleaning(targetDate) {
const threshold = this.hasTag('revered') ? 700 : 1500;
return this.daysSinceLastCleaning(targetDate) > threshold;
}
daysSinceLastCleaning(targetDate) {
return this._lastCleaned.until(targetDate, ChronoUnit.DAYS);
}
}
After
- id, title, hasTag와 같이 서브클래스에서 사용하는 슈퍼클래스의 동작 각각에 대응하는 전달 메서드를 만든다.
- 카탈로그 아이템 클래스 생성자를 추가하고 상속 관게를 끊는다.
class Scroll {
constructor(id, title, tags, dataLastCleaned) {
this._id = id;
this._catalogItem = new CatalogItem(id, title, tags);
this._lastCleaned = dataLastCleaned;
}
get id() { return this.id; }
get title() { return this._catalogItem.title; }
hasTag(aString) { return this._catalogItem.tags.hasTag(aString); }
needsCleaning(targetDate) {
const threshold = this.hasTag('revered') ? 700 : 1500;
return this.daysSinceLastCleaning(targetDate) > threshold;
}
daysSinceLastCleaning(targetDate) {
return this._lastCleaned.until(targetDate, ChronoUnit.DAYS);
}
}
수고하셨습니다 🎉ㅌ
반응형
'Programming > 13. Book' 카테고리의 다른 글
모던 리액트 Deep Dive - 12장 (0) | 2024.03.03 |
---|---|
리팩터링 - 7장 (1) | 2024.03.02 |
리팩터링 - 11장 (0) | 2024.03.02 |
리팩터링 - 10장 (0) | 2024.03.02 |
리팩터링 - 8, 9장 (0) | 2024.03.02 |
댓글