개요

다중 상속 언어가 아니면 일반적으로 상속은 단일 상속만 된다. 바꿔말하면 하나의 기준으로 분류할 밖에 없다. 상속은 부모클래스와 자식클래스가 강하게 묶이게된다. 부모를 수정하면 예외없이 모든 자식이 영향을 받는다.(단점이라는 소리는 아님)

 

위임은 이런 문제가 없다. 상속보다 결합도가 훨씬 낮기 때문이다.

 

디자인 원칙으로 "상속보다 구성을 사용하라"라는 말이 있다.

상속은 컴파일 시점에 행동이 고정된다. 반면에 구성은 런타임 시점에 동적으로 행동을 변화시킬 있다.

 

서브클래스를 위임으로 바꾸기 리팩터링은 디자인 패턴으로 따지면, 서브클래스를 상태나 전략 패턴으로 대체하기로 있다.

예시

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;
    }
}
// 클라이언트 (일반 예약)
aBooking = new Booking(show, date);
// 클라이언트 (프리미엄 예약)
aBooking = new PremiumBooking(show, date, extras);

서브 클래스를 위임으로 바꾸는 이유, 상속은 번만 가능하다. 다른 상속을 사용해야할 이유가 지금 상속 구조보다 크다면, 지금 상속은 구성으로 바꿔야한다.


//생성자를 팩터리 함수로 변경해 캡슐화
function createBooking(show, date){
    return new Booking(show, date);
}
function createPremiumBooking(show, date, extras){
    return new PremiumBooking(show, date, extras);
}




// 클라이언트 (일반 예약)
aBooking = createBooking(show, date);
// 클라이언트 (프리미엄 예약)
aBooking = createPremiumBooking(show, date, extras);


//위임 클래스 만들기
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        //서브클래스는 super 키워드로 부모 클래스 멤버에 쉽게 접근할 수 있지만,
        //위임에선 역참조가 필수다.
        this._host = hostBooking;
        this._extras =extras;
    }
}


//프리미엄 예약을 대체할 새로운 위임을 연결
function createPremiumBooking(show, date, extras){
    const result = new PremiumBooking(show, date, extras);
    result._bePremium(extras);
    return result;
}


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;
    }
    //이메서드가 private이라는 의미로 _ 붙임
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
}


class PremiumBooking extends Booking{
    constructor(show, date, extras){
        super(show, date);
        this._extras = extras;
    }
    get hasTalkback(){
        return this._premiumDelegate.hasTalkback;
    }
    get basePrice(){
        return Math.round(super.basePrice + this._extras.premiumFee);
    }
    get hasDinner(){
        return this._extras.hasOwnProperty('dinner') && !this.isPeakDay;
    }
}
//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
}


class PremiumBooking extends Booking{
    constructor(show, date, extras){
        super(show, date);
        this._extras = extras;
    }
    /* 위임 메서드가 잘 동작하면, 이제 제거
    get hasTalkback(){
        return this._premiumDelegate.hasTalkback;
    }*/
    get basePrice(){
        return Math.round(super.basePrice + this._extras.premiumFee);
    }
    get hasDinner(){
        return this._extras.hasOwnProperty('dinner') && !this.isPeakDay;
    }
}


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 basePrice(){
        let result = this._show.price;
        if(this.isPeakDay) result += Math.round(result * 0.15);
        return result;
    }
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
}
//문제 발생
class PremiumBooking extends Booking{
    constructor(show, date, extras){
        super(show, date);
        this._extras = extras;
    }
    get basePrice(){
        return this._premiumDelegate.basePrice;
    }
    get hasDinner(){
        return this._extras.hasOwnProperty('dinner') && !this.isPeakDay;
    }
}
//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
    //super로 부모 메서드 호출하는 부분을 방금과 같은 방식으로 리팩터링하면 무한 재귀에 빠진다., 상속을 구성으로 변경할 흔히 발생하는 문제
    get basePrice(){
        return Math.round(this._host.basePrice + this._extras.premiumFee);
    }
}


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;
    }
    //해결방안 1
    //슈퍼 클래스의 계산 로직을 함수로 추출, 가격 계산과 분배 로직 분리
    get basePrice(){
        return (this._premiumDelegate)
            ? this._premiumDelegate.basePrice
            : this._privateBasePrice;
    }
    get _privateBasePrice(){
        let result = this._show.price;
        if(this.isPeakDay) result += Math.round(result * 0.15);
        return result;
    }
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
}


//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
    get basePrice(){
        return Math.round(this._host._privateBasePrice + this._extras.premiumFee);
    }
}



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;
    }
    //해결방안 2
    // 위임 메서드를 기반 메서드의 확장 형태로 재호출
    get basePrice(){
        let result = this._show.price;
        if(this.isPeakDay) result += Math.round(result * 0.15);
        return (this._premiumDelegate)
            ? this._premiumDelegate.extendBasePrice(result)
            : result;
    }
   
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
}


//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
    extendBasePrice(base){
        return Math.round(base + this._extras.premiumFee);
    }
}


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 basePrice(){
        let result = this._show.price;
        if(this.isPeakDay) result += Math.round(result * 0.15);
        return (this._premiumDelegate)
            ? this._premiumDelegate.extendBasePrice(result)
            : result;
    }
   
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
    get hasDinner(){
        return (this._premiumDelegate)
            ? this._premiumDelegate.hasDinner
            : undefined;
    }
}
class PremiumBooking extends Booking{
    constructor(show, date, extras){
        super(show, date);
        this._extras = extras;
    }
}
//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
    extendBasePrice(base){
        return Math.round(base + this._extras.premiumFee);
    }


    get hasDinner(){
        return (this._premiumDelegate)
            ? this._premiumDelegate.hasDinner
            : undefined; //자바스크립트는 에러를 유연하게 처리하는 방식이 주류
            //다른 객체지향동적 언어라면, 예외를 던지는게 더 나은 선택일 수 있다.
    }
}


function createBooking(show, date){
    return new Booking(show, date);
}
function createPremiumBooking(show, date, extras){
    const result = new Booking(show, date, extras);
    result._bePremium(extras);
    return result;
}
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 basePrice(){
        let result = this._show.price;
        if(this.isPeakDay) result += Math.round(result * 0.15);
        return (this._premiumDelegate)
            ? this._premiumDelegate.extendBasePrice(result)
            : result;
    }
   
    _bePremium(extras){
        this._premiumDelegate = new PremiumBookingDelegate(this, extras);
    }
    get hasDinner(){
        return (this._premiumDelegate)
            ? this._premiumDelegate.hasDinner
            : undefined;
    }
}
//위임 클래스
class PremiumBookingDelegate{
    constructor(hostBooking, extras){
        this._host = hostBooking;
        this._extras =extras;
    }
    get hasTalkback(){
        return this._host._show.hasOwnProperty('talkback');
    }
    extendBasePrice(base){
        return Math.round(base + this._extras.premiumFee);
    }
    get hasDinner(){
        return this._extras.hasOwnProperty('dinner') && !this.isPeakDay;
    }
}
// 클라이언트 (일반 예약)
aBooking = createBooking(show, date);
// 클라이언트 (프리미엄 예약)
aBooking = createPremiumBooking(show, date, extras);

상속을 구성으로 바꾸면서, 분배 로직과 양방향 참조 추가 복잡도가 높아졌다. 그만큼 유연성을 증가해 런타임 동적으로 프리미엄 예약으로 바꿀 있다는 이점이 생겼다. 추가로 필요한 곳에 상속을 사용할 있게 됐다.

 

 

예시 : 서브클래스가 여러 개일



function createBird(data){
    switch(data.type){
        case '유럽 제비' :
            return new EuropeanSwallow(data);
        case '아프리카 제비' :
            return new AfricanSwallow(data);
        case '노르웨이 제비' :
            return new NorwegianBlueParrot(data);
        default : return new Bird(data);
    }
}
class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){return null;}
}
class EuropeanSwallow extends Bird{
    get airSpeedVelocity(){return 35;}
}
class AfricanSwallow extends Bird{
    constructor(data){
        super(data);
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class NorwegianBlueParrot extends Bird{
    constructor(data){
        super(data);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._plumage || "예쁘다";
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
}
야생 조류와 사육 조류를 구분 짓기
이렇게 구분 지을 경우, 기존 종에 따른 분류를 포기해야 한다.


//EuropeanSwallow부터 시작, 빈 위임 클래스 생성
class EuropeanSwallowDelegate{
}


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    //위임을 처리할 메서드 생성
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){return null;}
}


//위임 메서드로 옮김
class EuropeanSwallowDelegate{
    get airSpeedVelocity(){return 35;}
}
class EuropeanSwallow extends Bird{
    get airSpeedVelocity(){return this._speciesDelegate.airSpeedVelocity;}
}


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    //분배 메서드 처리
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}


//유럽 제비 클래스 제거, 관련 코드 정리
function createBird(data){
    switch(data.type){
        case '아프리카 제비' :
            return new AfricanSwallow(data);
        case '노르웨이 제비' :
            return new NorwegianBlueParrot(data);
        default : return new Bird(data);
    }
}


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data);
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}


//아프리카 제비 클래스 위임 처리
class AfricanSwallowDelegate{
    constructor(data){
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
}
class AfricanSwallow extends Bird{
    constructor(data){
        super(data);
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}


//메서드 옮기기
class AfricanSwallowDelegate{
    constructor(data){
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class AfricanSwallow extends Bird{
    constructor(data){
        super(data);
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return this._speciesDelegate.airSpeedVelocity;
    }
}


//아프리카 제비 클래스 제거, 관련 코드 정리
function createBird(data){
    switch(data.type){
        case '노르웨이 제비' :
            return new NorwegianBlueParrot(data);
        default : return new Bird(data);
    }
}
class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data);
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
class EuropeanSwallowDelegate{
    get airSpeedVelocity(){return 35;}
}
class AfricanSwallowDelegate{
    constructor(data){
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class NorwegianBlueParrot extends Bird{
    constructor(data){
        super(data);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._plumage || "예쁘다";
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
}


function createBird(data){
    switch(data.type){
        case '노르웨이 제비' :
            return new NorwegianBlueParrot(data);
        default : return new Bird(data);
    }
}
//노르웨이 제비도 같은 작업 수행
class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data);
            case '노르웨이 제비' :
                return new NorwegianBlueParrotDelegate(data);
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
class EuropeanSwallowDelegate{
    get airSpeedVelocity(){return 35;}
}
class AfricanSwallowDelegate{
    constructor(data){
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class NorwegianBlueParrotDelegate{
    constructor(data){
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
}
class NorwegianBlueParrot extends Bird{
    constructor(data){
        super(data);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._plumage || "예쁘다";
    }
    get airSpeedVelocity(){
        return this._speciesDelegate.airSpeedVelocity;
    }
}


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate();
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data);
            case '노르웨이 제비' :
                return new NorwegianBlueParrotDelegate(data,this);
            default: return null;
        }
    }
    get name(){return this._name;}
    get plumage(){return this._plumage || "보통이다";}
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
class NorwegianBlueParrotDelegate{
    //plumage() 옮기기, 오버라이드 메서드로 역참조 필요
    constructor(data, bird){
        this._bird = bird;//역참조
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._bird._plumage || "예쁘다";
    }
}
class NorwegianBlueParrot extends Bird{
    constructor(data){
        super(data);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get plumage(){
        return this._speciesDelegate.plumage;
    }
    get airSpeedVelocity(){
        return this._speciesDelegate.airSpeedVelocity;
    }
}
    //plumage() 분배 처리 문제 1
    get plumage(){
        if(this._speciesDelegate)
            return this._speciesDelegate.plumage;
        else
            return this._plumage || "보통이다";
    }
    //plumage() 분배 처리 문제 2
    //instanceof 연산자를 사용해 특정 클래스만 검사하는 것은 악취나는 코드다.
    get plumage(){
        if(this._speciesDelegate instanceof NorwegianBlueParrotDelegate)
            return this._speciesDelegate.plumage;
        else
            return this._plumage || "보통이다";
    }


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate(this);
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data,this);
            case '노르웨이 제비' :
                return new NorwegianBlueParrotDelegate(data,this);
            default: return null;
        }
    }
    get name(){return this._name;}
    //NorwegianBlueParrotDelegate만 특별 취급해보기
    get plumage(){
        if(this._speciesDelegate)
            return this._speciesDelegate.plumage;
        else
            return this._plumage || "보통이다";
    }
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
//plumage() 기본 메서드가 중복된다. 그리고 역참조를 추가하는 코드도 중복된다.
class EuropeanSwallowDelegate{
    constructor(bird){
        this._bird = bird;
    }
    get airSpeedVelocity(){return 35;}
    get plumage(){
        return this._bird._plumage || "예쁘다";
    }
}
class AfricanSwallowDelegate{
    constructor(data,bird){
        this._bird = bird;
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
    get plumage(){
        return this._bird._plumage || "예쁘다";
    }
}
class NorwegianBlueParrotDelegate{
    constructor(data, bird){
        this._bird = bird;
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._bird._plumage || "예쁘다";
    }
}


class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate(data, this);
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data,this);
            case '노르웨이 제비' :
                return new NorwegianBlueParrotDelegate(data,this);
            default: return new SpeciesDelegate(data, this);
        }
    }
    get name(){return this._name;}
    get plumage(){
        return this._speciesDelegate.plumage;
    }
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
//이 중복을 제거하기 위해 위임 클래스들에서 공통부분을 뽑아 슈퍼클래스로 만든다.
class SpeciesDelegate{
    constructor(data, bird){
        this._bird = bird;
    }
    get plumage(){
        return this._bird._plumage || "예쁘다";
    }
    airSpeedVelocity(){return null;}
}
class EuropeanSwallowDelegate extends SpeciesDelegate{
    constructor(data,bird){
        super(data, bird);
    }
    get airSpeedVelocity(){return 35;}
}
class AfricanSwallowDelegate extends SpeciesDelegate{
    constructor(data,bird){
        super(data, bird);
        this._bird = bird;
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class NorwegianBlueParrotDelegate extends SpeciesDelegate{
    constructor(data, bird){
        super(data,bird);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._bird._plumage || "예쁘다";
    }
}
//완료
function createBird(data){
    return new Bird(data);
}
class Bird{
    constructor(data){
        this._name = data.name;
        this._plumage = data.plumage;
        this._speciesDelegate = this.selectSpeciesDelegate(data);
    }
    selectSpeciesDelegate(data){
        switch(data.type){
            case "유럽 제비" :
                return new EuropeanSwallowDelegate(data, this);
            case "아프리카 제비" :
                return new AfricanSwallowDelegate(data,this);
            case '노르웨이 제비' :
                return new NorwegianBlueParrotDelegate(data,this);
            default: return new SpeciesDelegate(data, this);
        }
    }
    get name(){return this._name;}
    get plumage(){return this._speciesDelegate.plumage; }
    get airSpeedVelocity(){
        return (this._speciesDelegate)
            ? this._speciesDelegate.airSpeedVelocity
            : null;
    }
}
class SpeciesDelegate{
    constructor(data, bird){
        this._bird = bird;
    }
    get plumage(){
        return this._bird._plumage || "예쁘다";
    }
    airSpeedVelocity(){return null;}
}
class EuropeanSwallowDelegate extends SpeciesDelegate{
    constructor(data,bird){
        super(data, bird);
    }
    get airSpeedVelocity(){return 35;}
}
class AfricanSwallowDelegate extends SpeciesDelegate{
    constructor(data,bird){
        super(data, bird);
        this._bird = bird;
        this._numberOfCoconuts = data._numberOfCoconuts;
    }
    get airSpeedVelocity(){
        return 40 - 2 * this._numberOfCoconuts;
    }
}
class NorwegianBlueParrotDelegate extends SpeciesDelegate{
    constructor(data, bird){
        super(data,bird);
        this._voltage = data.voltage;
        this._isNailed = data.isNailed;
    }
    get airSpeedVelocity(){
        return( this._isNailed) ? 0 : 10 + this._voltage / 10;
    }
    get plumage(){
        if(this._voltage>100) return "그을렸다";
        else return this._bird._plumage || "예쁘다";
    }
}

 

+ Recent posts