Diana의 iOS 개발일기

[스위프트 프로그래밍 3판] - 17. 서브스크립트 본문

Swift/책 정리

[스위프트 프로그래밍 3판] - 17. 서브스크립트

Diana_iOS 2021. 3. 22. 13:56

서브스크립트는 인스턴스 이름 뒤에 값을 대괄호로 감싸 표현해줌으로써 값에 쉽게 접근하고 설정할 수 있습니다.

주로 각 타입의 구현부에 위치하며 매개변수로 다양한 값을 가질 수 있지만 입력 매개변수는 받지 못하는 등 쉬운 듯 하지만 꽤나 섬세한 문법을 가지고 있으므로 주의를 기울일 필요가 있습니다.


[서브스크립트 - Subscript]

서브스크립트는 컬렉션, 리스트, 시퀀스 등의 타입의 요소에 인덱스를 통해 접근하는 단축 문법입니다.

인덱스로 접근한다. 즉 우리가 평범하게 Array에 Array[Index], Dictionary에 Dictionary[Index]로 접근하던 것이 모두 서브스크립트입니다.

 

서브스크립트는 subscript 키워드를 사용하여 정의하며 인스턴스 메서드와 비슷하게 매개변수의 개수, 타입, 반환 타입 등을 지정할 수 있으며 기본 구현은 아래와 같습니다.

subscript(index: int) -> Int {
    get {
        //적절한 서브스크립트 결괏값 반환
    }
    
    set(newValue) { //newValue의 타입은 서브스크립트 반환 타입과 동일
        //적절한 설정자 역할 수행
    }
}

위의 구현을 보면 생성자(get)와 설정자(set)이 존재하고 설정자에 따로 매개변수를 지정해주지 않으면 암시적 전달인자 newValue를 사용할 수 있다는 점에서 연산 프로퍼티와 굉장히 흡사함을 알 수 있습니다.

서브스크립트는 또한 set 또는 get과 set 모두를 생략해줌으로써 읽기 전용의 구현이 가능합니다.

 

1. 서브스크립트의 구현

아래는 서브스크립트를 사용하여 구현한 예제입니다.

struct Student {
    var name: String
    var number: Int
}

class school {
    var number: Int = 0
    var student: [Student] = [Student]()
    
    func addStudent(name: String) {
        let student: Student = Student(name: name, number: self.number)
        self.students.append(student)
        self.number += 1
    }
    
    func addStudents(names: String...) {
        for name in names {
            self.addStudent(name: name)
        }
    }
    
    subscript(index: Int) -> Student? {
        if index < self.number {
            return self.students[index]
        }
        return nil
    }
}
let highSchool: School = School()
hightSchool.addStudents(names: "Diana", "Mia", "John", "Aile")

let aStudent: Student? = highSchool[1]
print("\(aStudent?.number) \(aStudent?.name)") //Optional(1) Optional("Mia")

 

2. 복수 서브스크립트

struct Student{
    var name: String
    var number: Int
}

class School {
    var number: Int = 0
    var students: [Student] = [Student]()
    
    func addStudent(name: String) {
        let student: Student = Student(name: name, number: self.number)
        self.students.append(student)
        self.number += 1
    }
    
    func addStudents(names: String...) {
        for name in names {
            self.addStudent(name: name)
        }
    }
    
    //첫 번째 서브스크립트
    subscript(index: Int) -> Student? { 
        get {
            if index < self.number {
                return self.students[index]
            }
            return nil
        }
        set {
            guard var newStudent: Student = newValue else {
                return
            }
            
            var number: Int = index
            
            if index > self.number {
                number = self.number
                self.number += 1
            }
            
            newStudent.number = number
            self.students[number] = newStudent
        }
    }
    
    //두 번째 서브스크립트
    subscript(name: String) -> Int? { 
        get {
            return self.students.filter{ $0.name == name }.first?.number
        }
        set {
            guard var number: Int = newValue else {
                return
            }
            
            if number > self.number {
                number = self.number
                self.number += 1
            }
            
            let newStudent: Student = Student(name: name, number: number)
            self.students[number] = newStudent
        }
    }
    
    //세 번째 서브스크립트
    subscript(name: String, number: Int) -> Student? { 
        return self.students.filter{ $0.name == name && $0.number == number }.first
    }
}

let highSchool: School = School()
hightSchool.addStudents(names: "Diana", "Mia", "John", "Aile")

let aStudent: Student? = highSchool[1]
print("\(aStudent?.number) \(aStudent?.name)") //Optional(1) Optional("Mia")

print(highSchool["Diana"])// Optional(0)
print(highSchool["Peter"]) //nil

highSchool[0] = Student(name: "Naiobe", number: 0)
highSchool["Amy"] = 1

print(highSchool["Mia"]) //nil
print(highSchool["Amy"]) //Optional(1)
print(highSchool["Aile", 3]) //Optional(Student(name: "Aile", number: 3))
print(highSchool["Paul", 3]) //nil

위와 같이 여러 개의 서브스크립트를 하나의 타입 안에 구현한 경우를 중복 정의라고 합니다.

 

3. 타입 서브스크립트

타입스크립트는 위에서 설명한 인스턴스에서 사용할 수 있는 서브스크립트와 달리, 타입 자체에서 사용할 수 있는 서브스크립트입니다. 타입스크립트를 사용하려면 subscript 키워드 앞에 static 키워드를 붙여야 합니다.

enum School: Int {
    case elementary = 1, middle, high, university
    
    static subscript(level: Int) -> School? {
        return Self(rawValue: level)
    }
}

let school: School? = School[2]
print(school) //School.middle

 

해당 글은 야곰님의 스위프트 프로그래밍 3판 을 기반으로 한 정리 글이며 문제가 있을 시 삭제하도록 하겠습니다.

스위프트 프로그래밍 3판 eBook 구매 링크: www.yes24.com/Product/Goods/81530016

 

스위프트 프로그래밍 (3판)

문법을 넘어 프로그래밍 패러다임도 익히는 스위프트 5스위프트 5의 핵심 키워드는 ‘안정화’다. ABI 안정화 덕분에 버전과 환경에 크게 영향받지 않고 더 유연하게 스위프트를 사용할 수 있게

www.yes24.com

야곰 님의 블로그: blog.yagom.net

 

yagom's blog

야곰의 프로그래밍 블로그입니다. iOS, Swift, Objective-C, C에 대해 이야기합니다.

blog.yagom.net