Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31
Archives
Today
Total
관리 메뉴

주니곰의 괴발노트

Swift - Concurrency (동시성) 본문

iOS

Swift - Concurrency (동시성)

Jhe2 2023. 2. 26. 16:11

Swift의 동시성

  • Swift 는 구조화된 방식으로 비동기 (asynchronous)병렬 (parallel) 코드 작성을 지원
  • 비동기 코드 (Asynchronous code) 는 일시적으로 중단되었다가 다시 실행할 수 있지만 한번에 프로그램의 한 부분만 실행
    • 프로그램에서 코드를 일시 중단하고 다시 실행하면 UI 업데이트와 같은 짧은 작업을 계속 진행하면서 네트워크를 통해 데이터를 가져오거나 파일을 분석하는 것과 같은 긴 실행 작업을 계속할 수 있음
  • 병렬 코드 (Parallel code) 는 동시에 코드의 여러부분이 실행됨을 의미
    • 예를 들어 4코어 프로세서의 컴퓨터는 각 코어가 하나의 작업을 수행하므로 코드의 4부분을 동시에 실행 가능
  • 병렬과 비동기 코드를 사용하는 프로그램은 한 번에 여러 작업을 수행
  • 외부 시스템을 기다리는 작업을 일시 중단하고 이 코드를 메모리 안전 방식으로 더 쉽게 작성할 수 있도록 함
  • 병렬 또는 비동기 코드의 추가 스케줄링 유연성에는 복잡성이 증가하는 비용도 수반
  • Swift 를 사용하면 일부 컴파일 시간 검사를 가능하게 하는 방식으로 표현할 수 있음
    • 예를 들어 실행자를 사용하여 변경가능한 상태에 안전하게 접근 가능하지만 느리거나 버그가 있는 코드에 동시성을 추가한다고 해서 코드가 빠르거나 올바르게 동작한다는 보장은 없음
    • 동시성을 추가하면 코드를 디버그하기 더 어렵게 만들 수도 있지만 그러나 동시성이 필요한 코드에서 동시성에 대한 Swift 의 언어 수준 지원을 사용하면 Swift 가 컴파일 시간에 문제를 찾는데 도움이 될 수 있음
  • Swift 에서 동시성 모델은 쓰레드의 최상단에 구축되지만 직접적으로 상호작용하지 않음
  • Swift 에서 비동기 함수는 실행중인 쓰레드를 포기할 수 있고, 첫번째 함수가 차단되는 동안 해당 쓰레드에서 다른 비동기 함수가 실행
  • 비동기 함수의 실행이 재개될 때 Swift 는 해당 함수가 실행될 쓰레드에 대해 어떠한 보장도 하지 않음

 

listPhotos(inGallery: "Summer Vacation") { photoNames in
    let sortedNames = photoNames.sorted()
    let name = sortedNames[0]
    downloadPhoto(named: name) { photo in
        show(photo)
    }
}
  • Swift 의 언어 지원을 사용하지 않고 동시성 코드를 작성할 수 있지만 해당 코드는 읽기가 더 어려운 경우가 있음
  • 위의 경우처럼 완료 핸들러가 연속해서 작성되어야 하므로 결국 중첩 클로저를 작성하게 됨
  • 이 스타일에서 더 많이 중첩된 복잡한 코드는 빠르게 다루기 어려울 수 있음
 

비동기 함수 정의와 호출

  • 비동기 함수 (asynchronous function) 또는 비동기 메서드 (asynchronous method) 는 실행 도중에 일시적으로 중단될 수 있는 특수한 함수 또는 메서드
  • 이것은 완료될 때까지 실행되거나 오류가 발생하거나 반환되지 않는 일반적인 동기 함수 또는 메서드 (synchronous functions and methods) 와 대조
  • 비동기 함수 또는 메서드는 이 세가지 중 하나를 수행하지만 무언가를 기다리고 있을 때 중간에 일시 중지될 수 있음
  • 비동기 함수 또는 메서드의 본문 내에서 실행을 일시 중지할 수 있는 부분을 표시
  • 함수 또는 메서드가 비동기 임을 나타내려면 던지는 함수 (throwing function) 를 나타내기 위해 throws 사용하는 것과 유사하게 파라미터 뒤의 선언에 async 키워드를 작성
  • 함수 또는 메서드가 값을 반환한다면 반환 화살표 (->) 전에 async 를 작성

 

func listPhotos(inGallery name: String) async -> [String] {
    let result = // ... some asynchronous networking code ...
    return result
}
  • 비동기와 던지기 둘 다인 함수 또는 메서드는 throws 전에 async 를 작성
  • 비동기 메서드를 호출할 때 해당 메서드가 반환될 때까지 실행이 일시 중단
  • 중단될 가능성이 있는 지점을 표시하기 위해 호출 앞에 await 를 작성
  • 이것은 에러가 있는 경우 프로그램의 흐름을 변경 가능함을 나타내기 위해 던지는 함수를 호출할 때 try 를 작성하는 것과 같음
  • 비동기 메서드 내에서 실행 흐름은 다른 비동기 메서드를 호출할 때만 일시 중단
    • 이것은 가능한 모든 중단 지점이 await 로 표시된다는 의미

 

let photoNames = await listPhotos(inGallery: "Summer Vacation")
let sortedNames = photoNames.sorted()
let name = sortedNames[0]
let photo = await downloadPhoto(named: name)
show(photo)
  • listPhotos(inGallery:) 와 downloadPhoto(named:) 함수 모두 네트워크 요청을 필요로 하기 때문에 완료하는데 비교적 오랜시간이 걸릴 수 있음
  • 반환 화살표 전에 async 를 작성하여 둘 다 비동기로 만들면 이 코드는 그림이 준비될 때까지 기다리는 동안 앱의 나머지 코드가 계속 실행될 수 있음
  • 위 동시성 예제 실행순서는 다음과 같음
    1. 코드는 첫번째 줄에서 실행을 시작하고 첫번째 await 까지 실행, listPhotos(inGallery:) 함수를 호출하고 해당 함수가 반환될 때까지 실행을 일시 중단
    2. 코드의 실행이 일시 중단되는 동안 같은 프로그램의 다른 동시 코드가 실행, 예를 들어 오랜시간 실행되는 백그라운드 작업이 새 사진을 리스트을 업데이트 할 수 있음. 이 코드는 await 로 표시된 다음 중단 지점 또는 완료될 때까지 실행
    3. listPhotos(inGallery:) 가 반환된 후에 이 코드는 해당 지점에서 시작하여 계속 실행, 반환된 값을 photoNames 에 할당
    4. sortedNames 와 name 을 정의하는 라인은 일반적인 동기 코드, 이 라인은 await 로 표시되지 않았으므로 가능한 중단 지점이 없음
    5. 다음 await 는 downloadPhoto(named:) 함수에 대한 호출을 표시. 이 코드는 해당 함수가 반환될 때까지 실행을 다시 일시 중단하여 다른 동시 코드에 실행할 기회를 제공
    6. downloadPhoto(named:) 가 반환된 후에 반환값은 photo 에 할당된 다음에 show(_:) 를 호출할 때 인수로 전달
 
 
  • await 로 표시된 코드의 중단이 가능한 지점은 비동기 함수 또는 메서드가 반환되기를 기다리는 동안 현재 코드 부분이 실행을 일시적으로 중단할 수 있음을 나타냄
  • Swift 가 현재 쓰레드에서 코드의 실행을 일시 중단하고 대신 해당 쓰레드에서 다른 코드를 실행하기 때문에 이것을 쓰레드 양보 (yielding the thread) 라고도 부름
  • await 가 있는 코드는 실행을 일시 중단할 수 있어야 하므로 다음과 같이 프로그램의 특정 위치에서만 비동기 함수 또는 메서드를 호출 가능
    • 비동기 함수, 메서드 또는 프로퍼티의 본문에 있는 코드
    • @main 으로 표시된 구조체, 클래스, 또는 열거형의 정적 (static) main() 메서드에 있는 코드
    • 구조화되지 않은 하위 작업의 코드
 
let firstPhoto = await listPhotos(inGallery: "Summer Vacation")[0]
add(firstPhoto toGallery: "Road Trip")
// At this point, firstPhoto is temporarily in both galleries.
remove(firstPhoto fromGallery: "Summer Vacation")
  • 중단이 가능한 지점 사이의 코드는 다른 비동기 코드의 중단 가능성없이 순차적으로 실행
  • add(_:toGallery:) 와 remove(_:fromGallery:) 사이에 실행되는 다른 코드는 없음
  • 그 시간동안 첫번째 사진은 양쪽 갤러리에 모두 나타나며 앱의 불변성 중 하나를 임시적으로 위반
  • 이 코드에 await 가 확실히 추가되지 말아야 한다는 것을 나타내기 위해 아래와 같이 동기 함수로 코드를 리팩토링 가능

 

func move(_ photoName: String, from source: String, to destination: String) {
    add(photoName, to: destination)
    remove(photoName, from: source)
}
// ...
let firstPhoto = await listPhotos(inGallery: "Summer Vacation")[0]
move(firstPhoto, from: "Summer Vacation", to: "Road Trip")
 
  • 위의 예제에서 move(_:from:to:) 함수는 동기 함수기 때문에 중단 가능한 지점을 포함히지 않는다는 것을 보장 가능
  • 이 함수에 중단 가능한 지점을 도입하기 위해 비동기 코드를 추가하면 버그가 아닌 컴파일 에러가 발생
 
func listPhotos(inGallery name: String) async throws -> [String] {
    try await Task.sleep(until: .now + .seconds(2), clock: .continuous)
    return ["IMG001", "IMG99", "IMG0404"]
}
  • Task.sleep(until:tolerance:clock:) 메서드는 동시성 작동 방식을 배우기 위해 간단한 코드를 작성할 때 유용
  • 이 메서드는 아무런 동작도 하지 않지만 반환되기 전에 주어진 나노 단위의 초만큼 기다림
  • 위 예제는 네트워크 작업 대기를 시뮬레이션하기 위해 sleep(until:tolerance:clock:) 을 사용하는 listPhoto(inGallery:) 함수의 버전

 

비동기 시퀀스

 

  • 이전 섹션에서 다뤘던 listPhotos(inGallery:) 함수는 비동기적으로 배열의 모든 요소가 준비된 후에 전체 배열을 한번에 반환
  • 또 다른 접근 방식은 비동기 시퀀스 (asynchronous sequence) 를 사용하여 한번에 콜렉션의 한 요소를 기다리는 것

 

import Foundation

let handle = FileHandle.standardInput
for try await line in handle.bytes.lines {
    print(line)
}

 

  • 비동기 시퀀스에 대한 조회 동작
  • 일반적인 for-in 루프 대신에 위의 예제는 for 다음에 await 를 작성
  • 비동기 함수 또는 메서드 호출할 때와 마찬가지로 await 작성은 가능한 중단 지점을 나타냄
  • for-await-in 루프는 다음 요소를 사용할 수 있을 때까지 기다리고 각 반복이 시작될 때 잠재적으로 실행을 일시 중단
  • Sequence 프로토콜에 준수성을 추가하여 for-in 루프에서 자체 타입을 사용할 수 있는 것과 같은 방식으로 AsyncSequence 프로토콜에 준수성을 추가하여 for-await-in 루프에서 자체 타입을 사용할 수 있음
 

 

비동기 함수의 병렬 호출

 

  • await 를 사용하여 비동기 함수를 호출하면 한번에 코드의 한 부분만 실행
  • 비동기 코드가 실행되는 동안 호출자는 코드의 다음 라인을 실행하기 위해 이동하기 전에 해당 코드가 완료될 때까지 기다림
    • 예를 들어 갤러리에서 처음 세 장의 사진을 가져오려면 다음과 같이 downloadPhoto(named:) 함수에 대한 세 번의 호출을 기다릴 수 있음

 

let firstPhoto = await downloadPhoto(named: photoNames[0])
let secondPhoto = await downloadPhoto(named: photoNames[1])
let thirdPhoto = await downloadPhoto(named: photoNames[2])

let photos = [firstPhoto, secondPhoto, thirdPhoto]
show(photos)

 

  • 이 방식은 다운로드가 비동기이고 진행되는 동안 다른 작업을 수행할 수 있지만 downloadPhoto(named:) 에 대한 호출은 한 번에 하나만 실행
  • 각 사진은 다음 사진이 다운로드를 시작하기 전에 완료
  • 그러나 이런 작업을 기다릴 필요없이 각 사진은 개별적으로 또는 동시에 다운로드 가능
 
async let firstPhoto = downloadPhoto(named: photoNames[0])
async let secondPhoto = downloadPhoto(named: photoNames[1])
async let thirdPhoto = downloadPhoto(named: photoNames[2])

let photos = await [firstPhoto, secondPhoto, thirdPhoto]
show(photos)

 

  • 비동기 함수를 호출하고 주변의 코드와 병렬로 실행하려면 상수를 정의할 때 let 앞에 async 를 작성하고 상수를 사용할 때마다 await 를 작성
  • 이 예제에서 downloadPhoto(named:) 을 호출하는 세가지는 모두 이전 호출이 완료되길 기다리지 않고 시작
  • 사용할 수 있는 시스템 자원이 충분하다면 동시에 실행 가능
  • 코드가 함수의 결과를 기다리기 위해 일시 중단되지 않기 때문에 이러한 함수 호출 중 어느 것도 await 로 표시하지 않고 photos 가 정의된 라인까지 실행이 계속됨
  • 이 시점에서 프로그램은 이러한 비동기 호출의 결과를 필요로 하므로 세 장의 사진이 모두 다운로드 될 때까지 실행을 일시 중단하기 위해 await 를 작성
  • 다음은 이 두 접근 방식의 차이점에 대해 생각할 수 있는 방법
    • 다음 줄의 코드가 해당 함수의 결과에 따라 달라지면 await 를 사용하여 비동기 함수를 호출. 이것은 순차적으로 실행되는 작업을 생성
    • 나중에 코드에서 결과가 필요하지 않을 때 async-let 을 사용하여 비동기 함수를 호출. 이렇게 하면 병렬로 수행할 수 있는 작업이 생성
    • await 와 async-let 은 모두 일시 중단되는 동안 다른 코드를 실행할 수 있도록 함
    • 두 경우 모두 비동기 함수가 반환될 때까지 필요한 경우 실행이 일시 중단됨을 나타내기 위해 가능한 일시 중단 지점을 await 로 표시
  • 동일한 코드에서 이 두 가지 접근 방식을 혼합할 수도 있음

 

작업과 작업 그룹

 

 

  • 작업 (task) 은 프로그램의 일부로 비동기적으로 실행할 수 있는 작업 단위
  • 모든 비동기 코드는 어떠한 작업의 일부로 실행
  • 이전 섹션에서 설명한 async-let 구문은 하위 작업을 생성
  • 작업 그룹 (task group) 을 생성하고 해당 그룹에 하위 작업을 추가할 수도 있으며 우선순위와 취소를 더 잘 제어할 수 있고 동적으로 작업의 수를 생성할 수 있음
  • 작업은 계층 구조로 정렬
  • 작업 그룹의 각 작업에는 동일한 상위 작업이 있으며 각 작업에는 하위 작업이 있을 수도 있음
  • 작업과 작업 그룹 간의 명시적 관계 때문에 이 접근 방식을 구조적 동시성 (structured concurrency) 이라고 함
  • 정확성에 대한 일부 책임을 지고 있지만 작업 간의 명시적 부모 (parent)-자식 (child) 관계를 통해 Swift 는 취소 전파 (propagating cancellation) 와 같은 일부 동작을 처리할 수 있고 Swift 는 컴파일 시간에 일부 오류를 감지 가능
 
await withTaskGroup(of: Data.self) { taskGroup in
    let photoNames = await listPhotos(inGallery: "Summer Vacation")
    for name in photoNames {
        taskGroup.addTask { await downloadPhoto(named: name) }
    }
}

 

구조화되지 않은 동시성

 

  • 이전 섹션에서 설명한 동시성에 대한 구조화된 접근 방식 외에도 Swift 는 구조화되지 않은 동시성 (unstructured concurrency) 을 지원
  • 작업 그룹의 일부인 작업과 달리 구조화되지 않은 작업 (unstructured task) 에는 상위 작업이 없음
  • 프로그램이 필요로 하는 방식으로 구조화되지 않은 작업을 관리할 수 있는 완전한 유연성이 있지만 정확성에 대한 완전한 책임도 있음
  • 현재 액터 (actor) 에서 실행되는 구조화되지 않은 작업을 생성하려면 Task.init(priority:operation:) 초기화 구문을 호출
  • 더 구체적으로 분리된 작업으로 알려진 현재 액터의 일부가 아닌 구조화되지 않은 작업을 생성하려면 Task.detached(priority:operation:) 클래스 메서드를 호출
  • 이 모든 동작은 서로 상호작용 할 수 있는 작업 (task)을 반환
    • 예를 들어 결과를 기다리거나 취소하는 경우
let newPhoto = // ... some photo data ...
let handle = Task {
    return await add(newPhoto, toGalleryNamed: "Spring Adventures")
}
let result = await handle.value

 

작업 취소

  • Swift 동시성은 협동 취소 모델 (cooperative cancellation model) 을 사용
  • 각 작업은 실행의 적절한 시점에서 취소되었는지 확인하고 적절한 방법으로 취소에 응답
  • 수행 중인 작업에 따라 일반적으로 다음 중 하나를 의미
    • CancellationError 와 같은 에러 발생
    • nil 또는 빈 콜렉션 반환
    • 부분적으로 완료된 작업 반환
       

 

  • 취소를 확인하려면 작업이 취소된 경우 CancellationError 를 발생시키는 Task.checkCancellation() 을 호출하거나 Task.isCancelled 의 값을 확인하고 자체 코드에서 취소를 처리
    • 예를 들어 갤러리에서 사진을 다운로드 하는 작업은 일부 다운로드를 삭제하고 네트워크 연결을 닫아야 할 수 있음
  • 취소를 수동으로 전파하려면 Task.cancel() 을 호출

 

액터

 

  • 프로그램을 동시성 조각으로 분리하기 위해 작업 (task) 을 사용할 수 있는데 작업은 서로 분리되어 있어 같은 시간에 안전하게 실행될 수 있지만 작업 간에 일부 정보를 공유해야 할 수도 있음
  • 액터 (Actors) 는 동시성 코드간에 정보를 안전하게 공유할 수 있게 해줌
  • 클래스와 마찬가지로 액터 (actors) 는 참조 타입이므로 클래스는 참조 타입 (Classes Are Reference Types) 에서 값 타입과 참조 타입의 비교는 클래스 뿐만 아니라 액터에도 적용
  • 클래스와 다르게 액터는 한 번에 하나의 작업만 변경 가능한 상태에 접근할 수 있도록 허용하므로 여러 작업의 코드가 액터의 동일한 인스턴스와 상호작용 하는 것은 안전

 

actor TemperatureLogger {
    let label: String
    var measurements: [Int]
    private(set) var max: Int
​
    init(label: String, measurement: Int) {
        self.label = label
        self.measurements = [measurement]
        self.max = measurement
    }
}
  • actor 키워드를 사용하여 액터를 도입하고 중괄호로 정의
  • TemperatureLogger 액터는 액터 외부의 다른 코드가 접근할 수 있는 프로퍼티가 있으며 액터 내부의 코드만 최대값을 업데이트 할 수 있게 max 프로퍼티를 제한
  • 구조체나 클래스와 같은 초기화 구문으로 액터의 인스턴스를 생성
 
let logger = TemperatureLogger(label: "Outdoors", measurement: 25)
print(await logger.max)
// Prints "25"
  • 액터의 프로퍼티 또는 메서드에 접근할 때 일시 중단 지점을 나타내기 위해 await 를 사용
  • 이 예제에서 logger.max 에 접근하는 것은 일시 중단 지점으로 가능
  • 액터는 한 번에 하나의 작업만 변경 가능한 상태에 접근할 수 있도록 허용하므로 다른 작업의 코드가 이미 로거와 상호 작용하고 있는 경우 이 코드는 프로퍼티 접근을 기다리는 동안 일시 중단
 
extension TemperatureLogger {
    func update(with measurement: Int) {
        measurements.append(measurement)
        if measurement > max {
            max = measurement
        }
    }
}
  • 대조적으로 액터의 부분인 코드는 액터의 프로퍼티에 접근할 때 await 를 작성하지 않음
  • update(with:) 메서드는 액터에서 이미 실행 중이므로 max 와 같은 프로퍼티에 대한 접근을 await 로 표시하지 않음
  • 이 메서드는 액터가 변경 가능한 상태와 상호 작용하기 위해 한 번에 하나의 작업만 허용하는 이유 중 하나를 보여줌
  • 액터의 상태에 대한 일부 업데이트는 일시적으로 불변성을 깨뜨림
  • TemperatureLogger 액터는 온도 리스트와 최대 온도를 추적하고 새로운 측정값을 기록할 때 최대 온도를 업데이트
  • 업데이트 도중에 새로운 측정값을 추가한 후 max 를 업데이트 하기 전에 온도 로거는 일시적으로 일치하지 않는 상태가 됨
  • 여러 작업이 동일한 인스턴스에 상호 작용하는 것을 방지하면 다음 이벤트 시퀀스와 같은 문제를 방지 가능
    • 코드는 update(with:) 메서드를 호출, 먼저 measurements 배열을 업데이트
    • 코드에서 max 를 업데이트 하기 전에 다른 코드에서 최대값과 온도 배열을 읽음
    • 코드는 max 를 변경하여 업데이트를 완료
  • 이러한 경우 다른 곳에서 실행 중인 코드는 데이터가 일시적으로 유효하지 않은 동안 update(with:) 호출 중간에 액터에 대한 접근이 인터리브 (interleaved) 되어 잘못된 정보를 읽음
  • Swift 액터는 한 번에 해당 상태에 대해 하나의 작업만 허용하고 해당 코드는 await 가 일시 중단 지점으로 표시되는 위치에서만 중단될 수 있기 때문에 Swift 액터를 사용하여 이 문제를 방지 가능
  • update(with:) 는 일시 중단 지점을 포함하지 않으므로 다른 코드는 업데이트 중간에 데이터에 접근 불가
  • 클래스의 인스턴스와 같이 액터의 외부에서 프로퍼티에 접근하려고 하면 컴파일 때 에러가 발생

 

print(logger.max)  // Error
  • await 작성 없이 logger.max 에 접근하는 것은 액터의 프로퍼티가 해당 액터의 분리된 로컬 상태의 부분이기 때문에 실패
  • Swift 는 액터 내부의 코드만 액터의 로컬 상태에 접근할 수 있도록 보장
  • 이 보장을 액터 분리 (actor isolation) 라고 표현

 

전송 가능 타입

 

  • 작업 (Tasks) 과 액터 (actors) 는 프로그램을 동시에 안전하게 실행할 수 있는 조각으로 나눌 수 있음
  • 작업 또는 액터의 인스턴스 내에서 변수와 프로퍼티와 같은 변경 가능한 상태를 포함하는 프로그램의 일부분을 동시성 도메인 (concurrency domain) 이라고 부름
  • 어떤 데이터는 데이터가 변경 가능한 상태를 포함하지만 동시 접근에 대해 보호되지 않으므로 동시성 도메인 간에 공유될 수 없음
  • 한 동시성 도메인에서 다른 동시성 도메인으로 공유될 수 있는 타입을 전송 가능 타입 (sendable type) 이라고 함
    • 예를 들어, 액터 메서드로 호출될 때 인수로 전달되거나 작업의 결과로 반환될 수 있음
  • 이 챕터의 앞부분에 있는 예제들은 동시성 도메인 간에 전달되는 데이터는 항상 안전한 간단한 값 타입을 사용했지만 일부 타입은 동시성 도메인 간에 전달하기 위해 안전하지 않음
    • 예를 들어, 변경 가능한 프로퍼티를 포함하고 해당 프로퍼티에 순차적으로 접근하지 않는 클래스는 서로 다른 작업 클래스의 인스턴스에 전달될 때 예상할 수 없고 잘못된 결과를 생성할 수 있음
  • Sendable 프로토콜을 선언하여 전송 가능한 타입으로 표시
  • 이 프로토콜은 어떠한 코드 요구사항을 가지지 않지만 Swift 가 적용하는 의미론적 요구사항이 있는데 다음과 같이 일반적으로 타입을 전송 가능한 것으로 나타내기 위한 세가지 방법이 있음
    • 타입은 값 타입이고 변경 가능한 상태는 다른 전송 가능한 데이터로 구성. Ex) 전송 가능한 저장된 프로퍼티가 있는 구조체 또는 전송 가능한 연관된 값이 있는 열거형
    • 타입은 변경 가능한 상태가 없으며 변경 불가능한 상태는 다른 전송 가능한 데이터로 구성. Ex) 읽기전용 프로퍼티만 있는 구조체 또는 클래스
    • 타입은 @MainActor 로 표시된 클래스나 특정 쓰레드나 큐에서 프로퍼티에 순차적으로 접근하는 클래스와 같이 변경 가능한 상태의 안정성을 보장하는 코드를 가지고 있음

 

struct TemperatureReading: Sendable {
    var measurement: Int
}

extension TemperatureLogger {
    func addReading(from reading: TemperatureReading) {
        measurements.append(reading.measurement)
    }
}

let logger = TemperatureLogger(label: "Tea kettle", measurement: 85)
let reading = TemperatureReading(measurement: 45)
await logger.addReading(from: reading)
  • 전송 가능한 프로퍼티만 가지는 구조체와 전송 가능한 연관된 값만 가지는 열거형과 같이 어떠한 타입은 항상 전송 가능
  • TemperatureReading 은 전송 가능한 프로퍼티만 가지는 구조체이며 public 또는 @usableFromInline 으로 표시되지 않은 구조체이므로 암시적으로 전송 가능

 

struct TemperatureReading {
    var measurement: Int
}
  • 위의 경우는 Sendable 프로토콜 준수가 암시되는 구조체

 

자료 출처

https://docs.swift.org/swift-book/documentation/the-swift-programming-language/concurrency

Comments