본문 바로가기
Swift 공식 가이드/Swift 2

Inheritance

by 토끼찌짐 2016. 4. 24.

Swift 3.0.1 가이드에 대응하는 정리글을 작성하였습니다!!!

Inheritance 정리 최신버전 링크 > http://wlaxhrl.tistory.com/46




Apple 제공 Swift 프로그래밍 가이드(2.2)의 Inheritance 부분을 공부하며 정리한 글입니다. 개인적인 생각도 조금 들어가있습니다.



들어가며

클래스는 다른 클래스로부터 메서드, 프로퍼티, 그 외 다른 특성들을 상속inheritance받을 수 있다. 상속을 받은 클래스를 서브클래스subclass, 상속을 해준 클래스를 슈퍼클래스superclass라고 부른다. 

서브클래스는 슈퍼클래스의 메서드, 프로퍼티, 스크립트에 접근할 수 있고, 재정의(오버라이딩)할 수도 있다.

상속받은 프로퍼티에 프로퍼티 옵저버를 추가할 수 있다. 지난번 프로퍼티 포스팅에 언급했던 것처럼, 슈퍼클래스의 computed 프로퍼티에도 서브클래스에서는 옵저버를 추가할 수 있다.



Defining a Base Class

베이스 클래스(Base Class) : 어떤 클래스도 상속받지 않은 클래스

<note> 스위프트에서는 모든 클래스가 상속받아야 할 universal base class가 특별히 없다. 따라서 아무 클래스도 상속받지 않은 클래스는 자연히 베이스 클래스가 된다.

class Vehicle {
    var currentSpeed = 0.0
    var description: String {
        return "현재 스피드는 \(currentSpeed)"
    }
    
    func makeNoise() {
        // 어떤 차일지에 따라 소리가 다르다
    }
}

let someVehicle = Vehicle()
print("Vehicle : \(someVehicle.description)")



Subclassing

서브클래싱은 기존 클래스(superclass)를 기반으로 하여 새로운 클래스(subclass)를 만드는 것이다. 서브클래스는 슈퍼클래스로부터 특성을 상속받고, 특성을 재정의할 수 있으며, 새로운 특성을 더할 수도 있다.

class SomeSubclass: SomeSuperclass {
    // subclass definition goes here
}


위에서 정의했던 베이스 클래스 Vehicle을 서브클래싱해보자.

class Bicycle: Vehicle {
    var hasBasket = false // 서브클래스만의 새로운 특성을 더했다
}

let bicycle = Bicycle()
bicycle.hasBasket = true
bicycle.currentSpeed = 15.0 // 슈퍼클래스의 프로퍼티를 상속받음
print(bicycle.description) // 슈퍼클래스의 메서드를 상속받음


// 서브클래스를 상속할 수도 있다
class Tandem: Bicycle {
    var currentNumberOfPassengers = 0
}

let tandem = Tandem()
print(tandem.description)
// Vehicle에 있던 decription 프로퍼티를 Bicycle이 상속받았고,
// Tandem이 Bicycle을 상속받았으므로 당연히 description에 접근 가능하다



Overriding

서브클래스는 슈퍼클래스의 메서드, 프로퍼티, 서브스크립트를 재정의할 수 있는데, 이것을 오버라이딩이라고 부른다. (따로 재정의하지 않으면 슈퍼클래스의 구현체를 그대로 사용한다)

오버라이딩을 위해서는 해당 메서드, 프로퍼티, 서브스크립트의 정의 앞에 override 키워드를 붙여준다.

  • override 키워드가 붙었는데 슈퍼클래스에 정의되어 있는 것과 인터페이스가 일치하지 않으면 에러

  • override 키워드가 안붙었는데 슈퍼클래스에 정의되어 있는 것과 인터페이스가 일치하면 에러 


<Accessing Superclass Methods, Properties, and Subscripts>

서브클래스 안에서 슈퍼클래스의 메서드, 프로퍼티, 서브스크립트를 오버라이드했더라도 경우에 따라서는 슈퍼클래스의 것을 사용해야 할 경우가 있다. 그럴때는 super 를 통해 접근한다.

  • super.someMethod()

  • super.somProperty

  • super[someIndex]


<Overriding Methods>

class Train: Vehicle {
    override func makeNoise() {
        print("칙폭칙폭")
    }
}

let train = Train()
train.makeNoise()


<Overriding Properties>

메서드와 마찬가지로 프로퍼티도 오버라이드할 수 있다. 또한 상속받은 프로퍼티에 옵저버를 추가할 수도 있다.


<Overriding Property Getters and Setters>

class Car: Vehicle {
    var gear = 1
    override var description: String {
        return super.description + " in gear \(gear)"
    }
}

상속받은 프로퍼티에 대해 custom getter/setter를 제공할 수 있다.

  • 즉, Computed 프로퍼티 형식으로 오버라이드하라는 것이다

  • 슈퍼클래스에서의 해당 프로퍼티가 Stored 프로퍼티여도 가능하다 (단 클래스 Stored 프로퍼티는 제외. 왜냐면 static 키워드를 붙여야만 정의할 수 있는데, static을 붙이면 overriding 할 수 없다.)

  • 서브클래스는 프로퍼티의 이름/타입만 알고 있으며 슈퍼클래스의 구현은 알지 못한다. 반대로 말하면 프로퍼티의 이름/타입을 명시해야 오버라이드 가능

  • 슈퍼클래스의 read-only(getter만 제공)프로퍼티를 서브클래스에서 read-writer(getter/setter 모두 제공)로 만드는 것은 가능하지만

  • 슈퍼클래스의 read-writer를 서브클래스에서 read-only로 만드는 것은 불가능. (즉, 이 경우는 getter만 오버라이드 불가능. getter/setter 모두 정의해주어야 한다) *부모가 가진 속성을 자식은 거부하면 안 된다(OOP 기본규칙 중 추상화 개념에 위배된다)

  • Computed 프로퍼티는 setter만 정의할 수 없는 것처럼, 프로퍼티 오버라이드의 경우에도 setter만 정의할 수 없다. getter도 반드시 같이 정의되어야 한다. 오버라이드 할 부분이 없다면 super.프로퍼티이름 이라도 넣어주어야 한다.


데...

stored 프로퍼티를 이런식으로 오버라이드하는 것은 왜 못하게 하는지... why?



<Overriding Property Observers>

프로퍼티 오버라이드를 사용하면 상속받은 프로퍼티에 옵저버를 추가할 수 있다.

class Vehicle {
    var currentSpeed = 0.0
    var description: String {
        return "현재 스피드는 \(currentSpeed)"
    }
    
    func makeNoise() {
        // 어떤 차일지에 따라 소리가 다르다
    }
}

// currentSpeed에 옵저버를 추가하기 위해 프로퍼티 오버라이드를 사용한다
class Car: Vehicle {
    var maxSpeed = 0.0
    override var currentSpeed: Double {
        willSet {
            if (newValue > maxSpeed) {
                maxSpeed = newValue
            }
        }
    }
}

var car = Car()
car.maxSpeed // 0
car.currentSpeed = 10.0
car.maxSpeed //10


// <추가설명> 이때 슈퍼클래스에도 currentSpeed에 옵저버가 걸려있다면
// 그것도 같이 불리게 된다. 아래 예시를 보자.

class Vehicle {
    var currentSpeed = 0.0 {
        willSet {
            print("willSet in super")
        }
    }
}

class Car: Vehicle {
    override var currentSpeed: Double {
        willSet {
            print("willSet in sub")
        }
    }
}

let car = Car()
car.currentSpeed = 3
// willSet in sub, willSet in super 둘 다 출력됨

// 위 상황에서 슈퍼클래스의 옵저버가 불리지 않게 하고 싶다면
// 프로퍼티의 setter, getter를 오버라이드 하면 된다고 하는데
// 여러가지로 시도해봤지만 상당히 코드가 더러워졌다
// 구글링을 해본 결과 다음과 같은 구조가 깔끔한 것 같다
// (http://stackoverflow.com/questions/29503482/override-property-observer)

class Vehicle {
    var currentSpeed = 0.0 {
        willSet {
            self.willCurrentSpeedChange()
        }
    }
    
    func willCurrentSpeedChange() {
        print("willSet in super")
    }
}

class Car: Vehicle {
    override func willCurrentSpeedChange() {
        print("willSet in sub")
    }
}


<note> 다음과 같은 예외사항들이 있다.

  • 상수 Stored 프로퍼티, getter만 있는 read-only Computed 프로퍼티에 옵저버를 추가하는 것은 불가. (변하지 않는 값이기 때문에 불필요)

  • 프로퍼티의 setter와 옵저버를 둘 다 오버라이드 하는 것은 불가능. (setter를 오버라이드 했다면 그 안에서 값의 변화를 탐지하도록 하자)



Preventing Overrides

서브클래스에서 오버라이드를 못하게 하고 싶다면 해당 메서드, 프로퍼티, 스크립트 정의 앞에 final 키워드를 붙인다.

'Swift 공식 가이드 > Swift 2' 카테고리의 다른 글

Initialization (2/3)  (0) 2016.05.06
Initialization (1/3)  (0) 2016.05.05
Subscripts  (0) 2016.04.23
Methods  (0) 2016.04.17
Properties  (1) 2016.04.13