You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
멀티 프로세스 환경에서 CPU가 어떤 하나의 프로세스를 실행하고 있는 상태에서, 인터럽트 요청에 의해 다음 순위의 프로세스가 실행되어야 할 때, OS 스케줄러가 기존의 프로세스의 상태 또는 레지스터 값(Context)을 저장하고 CPU가 다음 프로세스를 수행하도록 새로운 프로세스의 상태 또는 레지스터 값(Context)으로 교체하는 작업이다.
하이퍼 스레딩
가상의 두 번째 CPU를 인식시킨 다음 단일 CPU의 실행 유닛에 두 스레드를 번갈아 가며 실행하도록 하는 기법으로, 하나의 CPU를 두 개의 CPU인 것처럼 사용해 스레드를 동시에 처리하는 것을 말한다.
프로세스(process)
실행파일을 클릭했을 때, 메모리(RAM) 할당이 이루어지고, 이 메모리 공간으로 코드가 올라간다. 이 순간부터 이 프로그램은 '프로세스'라 불리게 된다. 즉, 사용자가 작성한 프로그램이 운영체제에 메모리 공간을 할당받아 실행 중인 상태이다.
운영체제로부터 시스템 자원을 할당받는 작업의 단위로 각각의 독립된 메모리 영역 (Code, Data, Stack, Heap)을 각자 할당 받아 안정적이다. 따라서 프로세스끼리는 서로의 변수나 자료구조에 대해 절대 접근할 수 없다.
Data, Resource, Thread로 구성
스레드(thread)
프로세스 내에서 실제 작업을 수행하는 흐름으로 모든 프로세스는 한 개 이상의 스레드가 존재하며, 두 개 이상의 스레드를 가지는 프로세스를 멀티 스레드 프로세스라고 한다. 물리적 스레드와 소프트웨어적 스레드가 있다.
쓰레드는 프로세스가 아닌, 프로세스 내에서 동작되는 것이기 때문에 메모리 영역을 독립적으로 할당받지 못한다. Code, Data, Heap 영역은 공유하고 Stack 영역만 독립적으로 할당하여 Task를 처리한다.
GCD
Sync와 Async
개별 스레드의 관점에서
Sync: Queue에 보낸 작업이 끝날 때까지 기다렸다가 다음 작업을 수행
Async: Queue에 작업을 보낸 직후 바로 다음 작업을 수행
Serial(Main)와 Concurrent(Global)
Queue에서의 작업 분배 관점에서
Serial(Main): Queue에서 작업을 분배하지 않고 대기열에 있는 작업을 다른 하나의 대상 스레드로 몰아서 순차적으로 작업, Main-thread 집중
Concurrent(Global): Queue에서 작업을 분배하여 여러 스레드로 동시 다발적으로 수행하도록 작업, Multi-thread 분산
Custom(Debug): 디버그 영역에서 확인 가능하도록 스레드에 대한 label을 부여하는 것이 가능
확인 사항
Main - async / Global - async
1. main 큐에서 다른 큐로 작업을 보낼 때 sync를 사용하면 안된다
Queue로 해당 작업을 보내어 할당하게 되면서, 다른 스레드로 Task를 동기적으로 분배한다. 하지만, 다른 스레드로 동기적(sync)으로 보내는 코드라도 결국, 실질적으로 기다렸다가 메인 스레드에서 작업 진행
2. 현재와 같은 큐에 sync로 작업을 보내면 안된다
//데드락 발생 가능성 있음
DispatchQueue.global().async{DispatchQueue.global().sync
}
//데드락 발생 가능성 없음
DispatchQueue.global(qos:.utility).async{DispatchQueue.global().sync
}
같은 큐에 보내게 되면 같은 스레드에 배치될 수 있는데, 해당 스레드가 sync 로 인해 멈춰있는 상황이라면 데드락 상황이 발생.
참고로 글로벌 큐는 Qos에 따라 각각 다른 큐 객체를 생성. 즉 DispatchQueue.global(qos: .utility) 와 DispatchQueue.global()는 다른 큐. 따라서 각각 다른 Qos 큐라면 쓰레드가 겹칠일이 없기 때문에, 데드락 발생 가능성이 없다.
3. main 스레드에서 DispatchQueue.main.sync를 사용하면 안된다
default로 main 스레드에서 작업하기에 Queue로 해당 작업을 보내어 할당하게 되면서, 무한정으로 완료를 기다리게 되므로 DeadLock(교착상태) 발생
4. QoS(GCD Quality Of Service)
Queue에서 작업 시작의 우선 순위를 정하여 분배하면서 어느정도 작업의 컨트롤이 가능, 시스템은 QoS정보를 사용하여 스케쥴링, CPU 및 I/O처리량 및 타이머 대기 시간과 같은 priority를 조정. 결과적으로 수행된 작업은 성능과 에너지 효율성 간의 균형을 유지한다. 하지만, 시작점은 컨트롤 할 수 있지만, 끝나는 시점에 대한 통제가 어렵다는
User-interactive : main thread에서 작업, 사용자 인터페이스(UI) 새로고침 또는 애니메이션 수행과 같이 사용자와 상호작용 하는 작업. 작업이 신속하게 수행되지 않으면, UI가 중단된 상태로 표시될 가능성이 있으므로 따라서, 반응성과 성능에 중점
User-initiated : 사용자가 시작한 작업이며, 저장된 문서를 열거나, 사용자 인터페이스에서 무언가를 클릭할 때 작업을 수행하는 것과 같은 즉각적인 결과가 필요. 사용자 상호작용을 계속하려면 작업이 필요합니다. 반응성과 성능에 중점
Utility : 작업을 완료하는 데 약간의 시간이 걸릴 수 있으며, 데이터 다운로드 또는 import와 같은 즉각적인 결과가 필요하지 않습니다. 작업은 초, 분 단위로 소요되며, 유틸리티 작업에는 일반적으로 사용자가 볼 수 있는 progress bar가 있습니다. 반응성, 성능 및 에너지 효율성 간에 균형을 유지하는 데 중점
Background : 백그라운드에서 작동하며, indexing, 동기화 및 백업과 같이 사용자가 볼 수 없는 작업. 분 또는 시간 단위로 소요되며, 에너지 효율성에 중점
클로저에서의 객체에 대한 캡처와 컴플리션 핸들러 주의
동작해야 할 task 를 queue에 보낸다는것은 결국 클로저를 통해 보내는 것. 따라서 객체에 대한 캡처 현상 발생할 수 있게 되고, 자칫하면 순환 참조가 생길 수도 있다. 따라서 앞서 배운 약한/미소유 참조를 활용하여 이를 예방해야한다
해당 queue에 보낸 task의 완료 시점에 대해서는 completion handler를 통하여 task별로 받는 방법이 있으나, 병렬적으로 관리와 noti가 가능한 DispatchGroup을 활용하는 것이 편리하다.
DispatchGroup
서로 다른 Task의 그룹화를 통하여 Queue에 보낸 작업들을 완료할 때 까지 기다리고, 완료 시 notification을 받도록 구성 가능하다. 여러 스레드로 분배된 작업들의 종료 시점을 각각이 아닌 하나로 그룹지어서 한번에 파악 가능!
1. group
내부적으로 비동기 활용 코드가 없는 경우 async 키워드에 group을 설정하면서 task를 바라보고 notification을 받을 수 있다. 하지만, 내부 네트워크 통신과 같은 백드라운드 비동기 함수가 포함되는 경우, 중복적으로 쓰레드 관련 제어 구문이 들어가기에 기대했던 정상적인 notify를 받을 수 없다.
notify 외에도 해당 그룹의 모든 작업이 완료때까지 현재 스레드를 block 시킬 수 있다. group1.wait(timeout:) 처럼 timeout 파라미터를 통해 얼마나 기다릴지에 대한 시간을 지정할 수 있다. 이를 통해 일정 시간 이후에는 다음 작업이 마저 진행된다. (그렇다고 시간내에 안끝난 작업을 멈추는건 아니고, 다른 스레드에서 계속 진행)
letgroup1=DispatchGroup()DispatchQueue.global().async(group: group1){}DispatchQueue.global().async(group: group1){}DispatchQueue.global(qos:.utility).async(group: group1){}lettimeoutResult= group1.wait(timeout:.now()+60)switch timeoutResult {case.success:print("60초 안에 그룹 내 모든 task 끝냄")case.timedOut:print("60초 안에 그룹 내 모든 task 못끝냄")}
wait() 함수를 실행하는 곳과 task를 보내는 큐 유의!
wait() 함수를 실행하는 곳이 메인스레드이면 안된다!
group1.wait() 를 통해 DispatchGroup에 대해 wait를 걸게 되는데, 여기서 주의할 점은 해당 함수를 실행하는 곳이 main큐(메인스레드)면 안된다. 왜냐하면 wait는 함수를 호출한 현재의 스레드를 블럭하기 때문에, 메인 스레드에서 실행하면 메인 스레드가 멈추는, 즉 task들이 다른 스레드에서 실행되는 시간만큼 앱이 멈추는 상황이 발생
그룹 내의 task는 wait가 실행되고 있는 현재의 스레드로 할당이 되어선 안된다.
왜냐하면,wait 를 실행하고 있는 스레드는 멈춰있을텐데 그 곳으로 task 가 할당된다면 데드락 상황이 발생한다. 현재의 스레드로 task를 할당 할 가능성이 있는 큐, 즉 현재 큐(현재 스레드에 wait을 실행하도록 할당한)로 task 를 보내면 안된다.
3. Enter / leave
내부적으로 비동기 활용 코드(animate, URLSession 등)가 있는 경우, Enter / leave를 통하여 Task의 완료 시점을 ARC를 통해서 참조를 관리하는 것과 유사하게 task reference count를 통한 관리가 가능하다.
다른 쓰레드에 같은 공유 자원에 접근하게 되는 경우 결과가 그때 그때 다르게 나올 수 있다. race condition은 random하게 발생하기도 해, 디버깅이 매우 어렵다. random하다는 것은 버그를 다시 재현하는 데에 명확한 과정이 없다는 것을 의미하기도 한다.
따라서, 동일한 데이터 접근 시에 경쟁하는 상황을 확인하고 방지하기 위해서 Scheme에서 Thread Sanitizer 옵션을 통해서 Debug 확인하여 이에 대하여 대응할 수 있다. race condition을 완벽히 피하는 방법은 없지만 queuing이나 GCD를 사용하는 등의 테크닉을 통해 훨씬 안전한 코드를 작성할 수 있다.
물론 이것 또한 실제적인 thread-safe한 방식을 적용에 있어서도 조금더 디테일한 컨트롤이 필요하다. 비동기 코드를 작성할 때 해당 코드들이 동시에(concurrently) 호출되었을 때 어떻게 동작할지 생각해보자!
serial queue + sync 조합의 엄격한 thread-safe 처리
concurrent queue + Dispatch Barrier
DispatchWorkItem
클로저 안에 넣어서 원하는 작업을 처리하는 것과 달리 class로 캡슐화하여 해당 task를 관리하고 싶어진다면 DispatchWorkItem 구문을 활용하여 보다 관리에 있어서 용이성을 높일 수 있다. 기존의 구문과 마찬가지로 Qos를 선정하여 task의 우선 순위를 지정하는 것도 가능하다. 이렇게 정의된 DispatchWorkItem 은 async(execute:) 라는 DispatchQueue의 instance method를 통해 큐에 보낼 수 있다. perform() 메소드를 통해 현재 스레드에서 sync 하게 동작도 가능하다.
// 다른 스레드를 활용한 비동기적 구성
DispatchQueue.global().async(execute: utilityItem)
// 현재 스레드를 활용한 동기적 구성
utilityItem.perform()
1. cancel
작업 실행 전
cancel을 요청하면 대기 중인 Queue에서 task가 제거된다.
작업 실행 중
작업이 멈추지는 않고 DispatchWorkItem 의 속성인 inCancelled 가 true 로 설정된다.
2. notify(queue:execute:)
작업 A가 끝난 후 작업 B가 특정 queue에서 실행(execute)되도록 지정할 수 있다.
Race Condition에서 보았듯 메모리 환경에는 공유자원이라는 개념이 있다. 공유자원을 안전하게 관리하기 위해서는 상호배제(Mutual exclusion)를 달성하는 기법이 필요하다. 물론 완벽하게 데이터 무결성을 보장하는 것은 힘들지만 GCD에서는 이에 대한 방안을 제공한다.
쉽게 말하면 정수 변수로서, 멀티프로그래밍 환경에서 공유 자원에 대한 접근을 제한하는 방법으로 공유 자원에 접근하는 작업의 수를 제한할 때 사용한다.
1. 동시 작업의 갯수 제한
iOS 에서는 세마포어를 위해 DispatchSemaphore 라는 객체를 이용한다. 공유 자원에 접근 가능한 (혹은 한번에 실행 가능한) 작업 수를 명시하고, 임계 구역에 들어갈때는 semaphore의 wait()를, 나올때는 signal()을 호출한다.
letsemaphore=DispatchSemaphore(value:2)foriin1...3{
semaphore.wait() // semaphore 감소
DispatchQueue.global().async(){
// 임계 구역(critical section)
print("공유 자원 접근 시작 \(i) 🌹")sleep(3)print("공유 자원 접근 종료 \(i) 🥀")
semaphore.signal() //semaphore 증가
}}
// 공유 자원 접근 시작 1 🌹
// 공유 자원 접근 시작 2 🌹
// 공유 자원 접근 종료 2 🥀
// 공유 자원 접근 종료 1 🥀
// 공유 자원 접근 시작 3 🌹
// 공유 자원 접근 종료 3 🥀
2. 두 스레드의 특정 이벤트 완료 상태 동기화
두 스레드가 특정 이벤트의 완료 상태를 동기화 하는 경우에 유용하다는 것입니다.
스레드 A는 작업 A 실행 중
스레드 B는 작업 A가 끝난 후에 무언가를 실행하려고 함
스레드 B(소비자)는 예상된 작업을 기다리기 위해 wait를 호출하고, 스레드 A(생성자)는 작업이 준비되면 signal를 호출하면 스레드 B가 작업 A의 완료 상태를 동기화할 수 있다. DispatchSemaphore를 해당 용도로 사용할때는 초기값을 0으로 설정한다.
// DispatchSemaphore 초기값 0으로 설정
letsemaphore=DispatchSemaphore(value:0)print("task A가 끝나길 기다림")
// 다른 스레드에서 task A 실행
DispatchQueue.global(qos:.background).async{
// task A
print("task A 시작!")print("task A 진행 중")print("task A 끝!")
// task A 끝났다고 알려줌
semaphore.signal()}
// task A 끝날 때까지는 value 가 0이라, task A 종료까지 block
semaphore.wait()print("task B 완료됨")
task A가 끝나지 않았다면 (즉 signal() 이 실행되지 않았다면) semaphore.wait() 이 후의 작업은 실행되지 않는다. 왜냐면 그전까지 세마포어 값은 0이기 때문이다.
NSOperationQueue
GCD는 우리가 Queue에 작업을 보내면 그에 따른 스레드를 적절히 생성해서 분배해주는 방법이다. Operation에서 사용하는 queue의 이름은 Operation Queue이며, 사실 내부적으론 GCD 위에서 동작한다. 특징적인 부분은 다음과 같다
동시에 실행할 수 있는 동작의 최대 수 지정
동작 일시 중지 및 취소
Async / Await
Swift 5.5에서 구현되어 기존의 비동기 처리의 문제를 해결하기 위해 등장하였다.
Swift 개발에서 Closure 및 completion handlers를 사용하는 asynchronous(비동기) 프로그래밍을 많이 함
많은 비동기 작업 / 오류 처리 / 비동기 호출간의 제어 흐름이 복잡할 때(스위즐링) 문제가 됨
Swift.Result가 Swift 5.0에서 추가되어 편리하지만 여전한 클로저 중첩 문제
위의 문제를 해결하기 위해 swift에 코루틴 동시 실행 설계 패턴을 도입(공식문서)
비동기 코드를 마치 동기 코드인것 처럼 작성 할 수 있음. ➞ 프로그래머가 동기 코드에서 사용할 수 있는 동일한 언어 구조를 최대한 활용 가능
자연스럽게 코드의 의미 구조를 보존 ➞ 언어에 대한 흐름성 방해에 대한 3가지의 주요 개선 제공
비동기 코드의 성능 향상(better performance for asynchronous code)
코드를 디버깅, 프로파일링 및 탐색하는 동안 보다 일관된 경험을 제공하기 위한 더 나은 도구 (better tooling to provide a more consistent experience while debugging, profiling, and exploring code)
작업 우선 순위 및 취소와 같은 동시성 기능을 위한 기반. (a foundation for future concurrency features like task priority and cancellation)
멀티스레딩을 위한 API라는 점에선 동일하나 GCD Queue는 복잡하지않고 가볍기 때문에 매우 간단한 동시성을 사용할 수 있다. 반면, NSOperationQueue는 GCD와 비교했을땐 추가적인 오버해드가 있으나, 다양한 작업들 가운데 의존성을 추가할 수 있고, 재사용, 취소, 중지시킬 수 있다.
Q. GCD API 동작 방식과 필요성에 대해 설명하시오.
Task를 Operation으로 Wrapping한 다음에, Queue에 넣는다. Queue에서 main 스레드 혹은 global 스레드에 작업을 배분한다. 동기, 비동기를 지정하여 조정한다.
웹에서 이미지를 다운 받아서 사용자에게 보여준다고 했을 때, 비동기로 처리하지 않는다면 이미지를 다운받는 동안 다른 작업을 할 수 없기 때문에 앱이 멈춘다. 이렇게 비용이 많이 들어가는 작업을 메인 스레드에서 진행하면 사용자가 다른 작업을 할 수 없기 때문에 필요하다고 생각한다.
기존에 스레드를 사용하려면 개발자가 직접 스레드를 생성하고 관리해야 했다. GCD를 사용하면 스레드 생성, 유지, 삭제 등을 개발자가 신경쓸 필요 없이 해야할 작업(코드)를 큐에 예약하기만 하면 된다.
Q. Global DispatchQueue 의 Qos 에는 어떤 종류가 있는지, 각각 어떤 의미인지 설명하시오
Qos는 중요한 순으로 userInteractive, userInitiated, default, utility, background, unspecified 가 있다.
우선순위가 더 높은 큐의 작업은 우선적으로 더 많은 쓰레드에 배치되고 배터리를 집중적으로 소모하게 된다.
userInteractive: main thread에서 작업. 사용자 인터페이스 새로고침 또는 애니메이션 수행과 같이 사용자와 상호작용을 하는 작업일때 사용한다. 순식간에 끝난다.
userInitiated: 사용자가 시작한 작업이며, 저장된 문서를 열거나 사용자 인터페이스에서 무언가를 클릭할 때 작업을 수행하는 것과 같은 즉각적인 결과가 필요할때 사용한다. 몇초 또는 그 이하에 끝난다.
default: userInitiated와 utility 사이의 중요도로, 개발자가 작업을 분류하는데 사용하기 위한것이 아니다. Qos가 할당되지 않은 작업은 default로 처리되며 GCD global queue는 이 레벨에서 실행된다.
utility: 작업을 완료하는 데 약간의 시간이 걸릴 수 있으며 데이터 다운로드 또는 import와 같이 즉각적인 결과가 필요하지 않은 작업, 응답성, 성능 및 에너지 효율성 간의 균형을 제공하는데 중점을 둔다.
background: 색인 생성, 동기화 및 백업과 같이 백그라운드에서 작동하고 사용자에게 표시되지 않는 작업이다. 에너지 효율성에 중점을 둔다.
unspecified: Qos가 없음을 나타내며 시스템에 Qos를 추론해야 한다는 신호를 보낸다.
Thread safe
동시성은 프로그래머가 맞이할 가장 복잡하고 기이한 버그의 시작점. 응용 프로그램 수준에서는 스레드와 하드웨어를 실제로 제어할 수 없기 때문에 멀티 스레드에서 동시에 사용할 때 시스템이 올바르게 동작하도록 보장하는 단위 테스트를 수행하기 어렵다.
스레드 안전성은 여러 스레드가 동시에 사용하려고 할 때 "정확성"을 보장하는 클래스의 기능, 각각의 다른 스레드가 어느 상태 값을 공유하려할 때 충돌이 나지 않는 상태가 스레드 세이프한 상태이며, 만약 충돌이 발생할 경우 OS 상의 동기화 API를 사용하여 예상가능하고 올바른 방식으로 보완이 가능
Thread Safety costs
어떤 동기화 작업을 수행하더라도 여기에 대한 performance hit가 발생하기 마련이다. 개발자는 절충을 통하여 유닛테스트가 불가능하고 예측 불가능한 상황이 벌어질 수 있는 상황을 미연에 방지할 수 있다. 때문에 병렬로 스레드가 접근하는 형태를 피하고 직렬적인 형태로 되도록 구성하는 것이 좋으며, 병렬로 구성할 때에는 각각의 시나리오를 잘 구성하여 충돌을 예방해야한다.
The text was updated successfully, but these errors were encountered:
동시성
컨텍스트 스위칭
멀티 프로세스 환경에서 CPU가 어떤 하나의 프로세스를 실행하고 있는 상태에서, 인터럽트 요청에 의해 다음 순위의 프로세스가 실행되어야 할 때, OS 스케줄러가 기존의 프로세스의 상태 또는 레지스터 값(Context)을 저장하고 CPU가 다음 프로세스를 수행하도록 새로운 프로세스의 상태 또는 레지스터 값(Context)으로 교체하는 작업이다.
하이퍼 스레딩
가상의 두 번째 CPU를 인식시킨 다음 단일 CPU의 실행 유닛에 두 스레드를 번갈아 가며 실행하도록 하는 기법으로, 하나의 CPU를 두 개의 CPU인 것처럼 사용해 스레드를 동시에 처리하는 것을 말한다.
프로세스(process)
실행파일을 클릭했을 때, 메모리(RAM) 할당이 이루어지고, 이 메모리 공간으로 코드가 올라간다. 이 순간부터 이 프로그램은 '프로세스'라 불리게 된다. 즉, 사용자가 작성한 프로그램이 운영체제에 메모리 공간을 할당받아 실행 중인 상태이다.
운영체제로부터 시스템 자원을 할당받는 작업의 단위로 각각의 독립된 메모리 영역 (Code, Data, Stack, Heap)을 각자 할당 받아 안정적이다. 따라서 프로세스끼리는 서로의 변수나 자료구조에 대해 절대 접근할 수 없다.
스레드(thread)
프로세스 내에서 실제 작업을 수행하는 흐름으로 모든 프로세스는 한 개 이상의 스레드가 존재하며, 두 개 이상의 스레드를 가지는 프로세스를 멀티 스레드 프로세스라고 한다. 물리적 스레드와 소프트웨어적 스레드가 있다.
쓰레드는 프로세스가 아닌, 프로세스 내에서 동작되는 것이기 때문에 메모리 영역을 독립적으로 할당받지 못한다. Code, Data, Heap 영역은 공유하고 Stack 영역만 독립적으로 할당하여 Task를 처리한다.
GCD
Sync와 Async
개별 스레드의 관점에서
Serial(Main)와 Concurrent(Global)
Queue에서의 작업 분배 관점에서
Custom(Debug): 디버그 영역에서 확인 가능하도록 스레드에 대한 label을 부여하는 것이 가능
확인 사항
1. main 큐에서 다른 큐로 작업을 보낼 때 sync를 사용하면 안된다
Queue로 해당 작업을 보내어 할당하게 되면서, 다른 스레드로 Task를 동기적으로 분배한다. 하지만, 다른 스레드로 동기적(sync)으로 보내는 코드라도 결국, 실질적으로 기다렸다가 메인 스레드에서 작업 진행
2. 현재와 같은 큐에 sync로 작업을 보내면 안된다
같은 큐에 보내게 되면 같은 스레드에 배치될 수 있는데, 해당 스레드가 sync 로 인해 멈춰있는 상황이라면 데드락 상황이 발생.
참고로 글로벌 큐는 Qos에 따라 각각 다른 큐 객체를 생성. 즉 DispatchQueue.global(qos: .utility) 와 DispatchQueue.global()는 다른 큐. 따라서 각각 다른 Qos 큐라면 쓰레드가 겹칠일이 없기 때문에, 데드락 발생 가능성이 없다.
3. main 스레드에서 DispatchQueue.main.sync를 사용하면 안된다
default로 main 스레드에서 작업하기에 Queue로 해당 작업을 보내어 할당하게 되면서, 무한정으로 완료를 기다리게 되므로 DeadLock(교착상태) 발생
4. QoS(GCD Quality Of Service)
Queue에서 작업 시작의 우선 순위를 정하여 분배하면서 어느정도 작업의 컨트롤이 가능, 시스템은 QoS정보를 사용하여 스케쥴링, CPU 및 I/O처리량 및 타이머 대기 시간과 같은 priority를 조정. 결과적으로 수행된 작업은 성능과 에너지 효율성 간의 균형을 유지한다. 하지만, 시작점은 컨트롤 할 수 있지만, 끝나는 시점에 대한 통제가 어렵다는
User-interactive : main thread에서 작업, 사용자 인터페이스(UI) 새로고침 또는 애니메이션 수행과 같이 사용자와 상호작용 하는 작업. 작업이 신속하게 수행되지 않으면, UI가 중단된 상태로 표시될 가능성이 있으므로 따라서, 반응성과 성능에 중점
User-initiated : 사용자가 시작한 작업이며, 저장된 문서를 열거나, 사용자 인터페이스에서 무언가를 클릭할 때 작업을 수행하는 것과 같은 즉각적인 결과가 필요. 사용자 상호작용을 계속하려면 작업이 필요합니다. 반응성과 성능에 중점
Utility : 작업을 완료하는 데 약간의 시간이 걸릴 수 있으며, 데이터 다운로드 또는 import와 같은 즉각적인 결과가 필요하지 않습니다. 작업은 초, 분 단위로 소요되며, 유틸리티 작업에는 일반적으로 사용자가 볼 수 있는 progress bar가 있습니다. 반응성, 성능 및 에너지 효율성 간에 균형을 유지하는 데 중점
Background : 백그라운드에서 작동하며, indexing, 동기화 및 백업과 같이 사용자가 볼 수 없는 작업. 분 또는 시간 단위로 소요되며, 에너지 효율성에 중점
클로저에서의 객체에 대한 캡처와 컴플리션 핸들러 주의
동작해야 할 task 를 queue에 보낸다는것은 결국 클로저를 통해 보내는 것. 따라서 객체에 대한 캡처 현상 발생할 수 있게 되고, 자칫하면 순환 참조가 생길 수도 있다. 따라서 앞서 배운 약한/미소유 참조를 활용하여 이를 예방해야한다
해당 queue에 보낸 task의 완료 시점에 대해서는 completion handler를 통하여 task별로 받는 방법이 있으나, 병렬적으로 관리와 noti가 가능한 DispatchGroup을 활용하는 것이 편리하다.
DispatchGroup
서로 다른 Task의 그룹화를 통하여 Queue에 보낸 작업들을 완료할 때 까지 기다리고, 완료 시 notification을 받도록 구성 가능하다. 여러 스레드로 분배된 작업들의 종료 시점을 각각이 아닌 하나로 그룹지어서 한번에 파악 가능!
1. group
내부적으로 비동기 활용 코드가 없는 경우 async 키워드에 group을 설정하면서 task를 바라보고 notification을 받을 수 있다. 하지만, 내부 네트워크 통신과 같은 백드라운드 비동기 함수가 포함되는 경우, 중복적으로 쓰레드 관련 제어 구문이 들어가기에 기대했던 정상적인 notify를 받을 수 없다.
2. wait
notify 외에도 해당 그룹의 모든 작업이 완료때까지 현재 스레드를 block 시킬 수 있다. group1.wait(timeout:) 처럼 timeout 파라미터를 통해 얼마나 기다릴지에 대한 시간을 지정할 수 있다. 이를 통해 일정 시간 이후에는 다음 작업이 마저 진행된다. (그렇다고 시간내에 안끝난 작업을 멈추는건 아니고, 다른 스레드에서 계속 진행)
group1.wait() 를 통해 DispatchGroup에 대해 wait를 걸게 되는데, 여기서 주의할 점은 해당 함수를 실행하는 곳이 main큐(메인스레드)면 안된다. 왜냐하면
wait는 함수를 호출한 현재의 스레드를 블럭
하기 때문에, 메인 스레드에서 실행하면 메인 스레드가 멈추는, 즉task들이 다른 스레드에서 실행되는 시간만큼 앱이 멈추는 상황이 발생
왜냐하면,
wait 를 실행하고 있는 스레드는 멈춰있을텐데 그 곳으로 task 가 할당된다면 데드락 상황이 발생
한다. 현재의 스레드로 task를 할당 할 가능성이 있는 큐, 즉 현재 큐(현재 스레드에 wait을 실행하도록 할당한)로 task 를 보내면 안된다.3. Enter / leave
내부적으로 비동기 활용 코드(animate, URLSession 등)가 있는 경우, Enter / leave를 통하여 Task의 완료 시점을 ARC를 통해서 참조를 관리하는 것과 유사하게 task reference count를 통한 관리가 가능하다.
Race Condition
다른 쓰레드에 같은 공유 자원에 접근하게 되는 경우 결과가 그때 그때 다르게 나올 수 있다. race condition은 random하게 발생하기도 해, 디버깅이 매우 어렵다. random하다는 것은 버그를 다시 재현하는 데에 명확한 과정이 없다는 것을 의미하기도 한다.
따라서, 동일한 데이터 접근 시에 경쟁하는 상황을 확인하고 방지하기 위해서 Scheme에서 Thread Sanitizer 옵션을 통해서 Debug 확인하여 이에 대하여 대응할 수 있다. race condition을 완벽히 피하는 방법은 없지만 queuing이나 GCD를 사용하는 등의 테크닉을 통해 훨씬 안전한 코드를 작성할 수 있다.
물론 이것 또한 실제적인 thread-safe한 방식을 적용에 있어서도 조금더 디테일한 컨트롤이 필요하다. 비동기 코드를 작성할 때 해당 코드들이 동시에(concurrently) 호출되었을 때 어떻게 동작할지 생각해보자!
DispatchWorkItem
클로저 안에 넣어서 원하는 작업을 처리하는 것과 달리 class로 캡슐화하여 해당 task를 관리하고 싶어진다면 DispatchWorkItem 구문을 활용하여 보다 관리에 있어서 용이성을 높일 수 있다. 기존의 구문과 마찬가지로 Qos를 선정하여 task의 우선 순위를 지정하는 것도 가능하다. 이렇게 정의된 DispatchWorkItem 은 async(execute:) 라는 DispatchQueue의 instance method를 통해 큐에 보낼 수 있다. perform() 메소드를 통해 현재 스레드에서 sync 하게 동작도 가능하다.
1. cancel
cancel을 요청하면 대기 중인 Queue에서 task가 제거된다.
작업이 멈추지는 않고 DispatchWorkItem 의 속성인 inCancelled 가 true 로 설정된다.
2. notify(queue:execute:)
작업 A가 끝난 후 작업 B가 특정 queue에서 실행(execute)되도록 지정할 수 있다.
DispatchSemaphore
Race Condition에서 보았듯 메모리 환경에는 공유자원이라는 개념이 있다. 공유자원을 안전하게 관리하기 위해서는 상호배제(Mutual exclusion)를 달성하는 기법이 필요하다. 물론 완벽하게 데이터 무결성을 보장하는 것은 힘들지만 GCD에서는 이에 대한 방안을 제공한다.
👉🏻 뮤텍스(Mutex)와 세마포어(Semaphore)의 차이
쉽게 말하면 정수 변수로서, 멀티프로그래밍 환경에서 공유 자원에 대한 접근을 제한하는 방법으로 공유 자원에 접근하는 작업의 수를 제한할 때 사용한다.
1. 동시 작업의 갯수 제한
iOS 에서는 세마포어를 위해 DispatchSemaphore 라는 객체를 이용한다. 공유 자원에 접근 가능한 (혹은 한번에 실행 가능한) 작업 수를 명시하고, 임계 구역에 들어갈때는 semaphore의 wait()를, 나올때는 signal()을 호출한다.
2. 두 스레드의 특정 이벤트 완료 상태 동기화
두 스레드가 특정 이벤트의 완료 상태를 동기화 하는 경우에 유용하다는 것입니다.
스레드 B(소비자)는 예상된 작업을 기다리기 위해 wait를 호출하고, 스레드 A(생성자)는 작업이 준비되면 signal를 호출하면 스레드 B가 작업 A의 완료 상태를 동기화할 수 있다. DispatchSemaphore를 해당 용도로 사용할때는 초기값을 0으로 설정한다.
task A가 끝나지 않았다면 (즉 signal() 이 실행되지 않았다면) semaphore.wait() 이 후의 작업은 실행되지 않는다. 왜냐면 그전까지 세마포어 값은 0이기 때문이다.
NSOperationQueue
GCD는 우리가 Queue에 작업을 보내면 그에 따른 스레드를 적절히 생성해서 분배해주는 방법이다. Operation에서 사용하는 queue의 이름은 Operation Queue이며, 사실 내부적으론 GCD 위에서 동작한다. 특징적인 부분은 다음과 같다
Async / Await
Swift 5.5에서 구현되어 기존의 비동기 처리의 문제를 해결하기 위해 등장하였다.
위의 문제를 해결하기 위해 swift에 코루틴 동시 실행 설계 패턴을 도입(공식문서)
비동기 코드를 마치 동기 코드인것 처럼 작성 할 수 있음. ➞ 프로그래머가 동기 코드에서 사용할 수 있는 동일한 언어 구조를 최대한 활용 가능
자연스럽게 코드의 의미 구조를 보존 ➞ 언어에 대한 흐름성 방해에 대한 3가지의 주요 개선 제공
Q. NSOperationQueue 와 GCD Queue 의 차이점을 설명하시오.
멀티스레딩을 위한 API라는 점에선 동일하나 GCD Queue는 복잡하지않고 가볍기 때문에 매우 간단한 동시성을 사용할 수 있다. 반면, NSOperationQueue는 GCD와 비교했을땐 추가적인 오버해드가 있으나, 다양한 작업들 가운데 의존성을 추가할 수 있고, 재사용, 취소, 중지시킬 수 있다.
Q. GCD API 동작 방식과 필요성에 대해 설명하시오.
Task를 Operation으로 Wrapping한 다음에, Queue에 넣는다. Queue에서 main 스레드 혹은 global 스레드에 작업을 배분한다. 동기, 비동기를 지정하여 조정한다.
웹에서 이미지를 다운 받아서 사용자에게 보여준다고 했을 때, 비동기로 처리하지 않는다면 이미지를 다운받는 동안 다른 작업을 할 수 없기 때문에 앱이 멈춘다. 이렇게 비용이 많이 들어가는 작업을 메인 스레드에서 진행하면 사용자가 다른 작업을 할 수 없기 때문에 필요하다고 생각한다.
기존에 스레드를 사용하려면 개발자가 직접 스레드를 생성하고 관리해야 했다. GCD를 사용하면 스레드 생성, 유지, 삭제 등을 개발자가 신경쓸 필요 없이 해야할 작업(코드)를 큐에 예약하기만 하면 된다.
Q. Global DispatchQueue 의 Qos 에는 어떤 종류가 있는지, 각각 어떤 의미인지 설명하시오
Qos는 중요한 순으로 userInteractive, userInitiated, default, utility, background, unspecified 가 있다.
우선순위가 더 높은 큐의 작업은 우선적으로 더 많은 쓰레드에 배치되고 배터리를 집중적으로 소모하게 된다.
Thread safe
동시성은 프로그래머가 맞이할 가장 복잡하고 기이한 버그의 시작점. 응용 프로그램 수준에서는 스레드와 하드웨어를 실제로 제어할 수 없기 때문에 멀티 스레드에서 동시에 사용할 때 시스템이 올바르게 동작하도록 보장하는 단위 테스트를 수행하기 어렵다.
스레드 안전성은 여러 스레드가 동시에 사용하려고 할 때 "정확성"을 보장하는 클래스의 기능, 각각의 다른 스레드가 어느 상태 값을 공유하려할 때 충돌이 나지 않는 상태가 스레드 세이프한 상태이며, 만약 충돌이 발생할 경우 OS 상의 동기화 API를 사용하여 예상가능하고 올바른 방식으로 보완이 가능
Thread Safety costs
어떤 동기화 작업을 수행하더라도 여기에 대한 performance hit가 발생하기 마련이다. 개발자는 절충을 통하여 유닛테스트가 불가능하고 예측 불가능한 상황이 벌어질 수 있는 상황을 미연에 방지할 수 있다. 때문에 병렬로 스레드가 접근하는 형태를 피하고 직렬적인 형태로 되도록 구성하는 것이 좋으며, 병렬로 구성할 때에는 각각의 시나리오를 잘 구성하여 충돌을 예방해야한다.
The text was updated successfully, but these errors were encountered: