본문 바로가기

All

(44)
[iOS] 강한 참조 사이클 (순환 참조) 해결하기 2편 두 인스턴스가 서로를 참조할 경우 강한 참조 사이클이 발생하고 인스턴스가 메모리에서 정상적으로 해제되지 않는 것을 강한 참조 사이클 해결하기 1탄에서 확인하였습니다. 강한 참조 사이클(Strong Reference Cycle)이 발생하는 문제를 해결하기 위해 약한(weak) 참조와 비소유(unowned) 참조로 해결하였죠. 클로저(Closure)에서도 강한 참조 사이클이 발생할 수 있습니다. 클로저가 인스턴스를 캡처하고 인스턴스가 클로저를 강한 참조로 저장하고 있다면 인스턴스는 메모리에서 정상적으로 해제되지 않습니다. 이 경우에도 약한 참조나 비소유 참조를 통해 해결할 수 있습니다. 강한 참조 사이클이 발생하는 경우를 예시를 통해 살펴보겠습니다. class Car { var totalDrivingDist..
[iOS] 강한 참조 사이클 (순환 참조) 해결하기 1편 인스턴스를 사용하지 않는데도 메모리에서 해제 되지 않으면 메모리 누수(Memory Leak)가 발생합니다. 강한 참조 사이클(Strong Reference Cycle)에 의해 발생할 수 있으며 약한(Weak) 참조와 비소유(Unowned) 참조를 통해 해결할 수 있습니다. 강한 참조에 대한 자세한 설명은 ARC 란? 을 참고해주세요! 강한 참조 사이클은 순환 참조(Retain Cycle)라고도 불립니다. 예시를 통해 강한 참조 사이클이 발생하는 경우를 먼저 살펴보도록 하겠습니다. class Person { var name = "John" var apartment: Apartment? deinit { print("Person deinit") } } class Apartment { var address: S..
[iOS] ARC 란? (Automatic Reference Counting) 앱의 성능을 유지하기 위해서는 메모리 관리가 기본이자 필수적 입니다. 그렇기 때문에 코코아(Cocoa)에서 사용하는 메모리 관리모델을 이해할 필요가 있습니다. 바로 MRC와 ARC 2가지 관리 모델입니다. Cocoa란? 애플에서 제공하는 어플리케이션 개발 환경을 의미합니다. macOS에서는 Cocoa, iOS에서는 Cocoa Touch 라고 부릅니다. 스택(Stack) 메모리에 저장된 데이터는 자동으로 제거되기 때문에 특별한 관리가 필요없습니다. 하지만 힙(Heap) 메모리에 저장된 데이터는 필요하지 않은 시점에 직접 제거해야만 합니다. 메모리 관리 모델은 힙에 저장된 데이터를 관리합니다. Value Type: Structure, Enumeration, Tuple Reference Type: Class,..
[Swift] Type Casting (is, as, as?, as!) 타입캐스팅 완벽 정리 타입 캐스팅(Type Casting) 은 인스턴스의 타입을 확인하거나, 해당 인스턴스를 자신의 클래스 계층에 있는 상위 혹은 하위 클래스로 처리하는 방법입니다. 스위프트의 타입 캐스팅은 is 와 as 연산자로 구현됩니다. 이 두 연산자는 값의 타입을 확인하거나 값을 다른 타입으로 지정합니다. 단순하면서도 보고 이해하기 쉬운 표현 방식이죠. 또한, 타입 캐스팅을 사용하여 해당 타입이 프로토콜을 따르는지(Protocol Conformance)도 확인할 수 있습니다. 타입 캐스팅 연산자 타입 캐스팅 연산자는 총 4가지가 있습니다. expression is type expression as type expression as? type expression as! type is 연산자는 런타임에 expression..
[Swift] Escaping Closure 탈출 하는 클로저!? 클로저(Closure)가 함수의 인자로 전달되고 함수 외부에서 실행될 때 함수를 탈출(escape)하는 클로저라고 합니다. 이 클로저는 함수가 반환된 후에 호출됩니다. 함수의 매개변수 타입 앞에 @escaping을 작성하여 해당 클로저가 함수를 탈출할 수 있음을 나타냅니다. 클로저가 함수를 탈출할 수 있는 방법은 함수 외부에 정의된 변수에 저장되는 것입니다. 예를 들어, 비동기 작업을 해야하는 많은 함수들은 completion handler 매개변수로 클로저를 받습니다. 해당 클로저는 함수의 작업이 완료될 때 까지 호출되지 않습니다. 즉, completion handler 는 함수의 작업이 완료된 후 호출되는 것을 보장받을 수 있습니다. var completionHandlers: [() -> Void] ..
[Swift] Capturing Values 값을 캡쳐한다는 것 Swift에서 값을 캡쳐(Capture)한다는 것의 의미를 한 번 알아보겠습니다. 클로저(Closure)는 정의된 주변의 컨텍스트에 있는 상수나 변수들을 캡처할 수 있습니다. 그런 다음 클로저는 상수와 변수를 정의한 원래 범위가 더 이상 존재하지 않더라도 그 상수와 변수의 값을 참조하고 수정할 수 있습니다. 스위프트에서 값을 캡처할 수 있는 클로저의 가장 간단한 형태는 중첩 함수(Nested Function)로서, 다른 함수의 body 안에 정의되어 있습니다. 중첩 함수는 외부 함수의 인수를 캡처할 수 있으며, 또한 외부 함수 내에 정의된 상수와 변수를 캡처할 수 있습니다. incrementer 중첩 함수를 포함한 makeIncrementer 함수 예제를 살펴보겠습니다. incrementer 함수는 ru..
[Swift] Closure? 함수 블록 클로저에 대해 알아보자 클로져(Closure)란 코드에서 전달되고 사용될 수 있는 자체적인(self-contained) 함수 블록입니다. 스위프트에서의 클로저는 C와 Objective-C에서의 블록과 유사하고 다른 프로그래밍 언어에서는 람다와 유사합니다. 클로저는 클로저가 선언된 컨텍스트에 있는 모든 상수나 변수들의 참조값을 저장하고 캡처할 수 있습니다. 이것은 상수나 변수들을 closing over 한다고 표현하기도 합니다. 스위프트는 캡처에 대한 모든 메모리 관리를 다룹니다. 값을 캡처하는 것(Capturing Values)에 대한 개념은 다음 포스팅에서 자세히 다루도록 하겠습니다. 흔히 알고있는 글로벌 함수, 중첩 함수(함수 내부에 선언된 함수) 들은 실제로는 클로저의 특별한 케이스 입니다. 클로저는 다음 3가지 중 하나..
[iOS] GCD 활용하기 2편 (DispatchGroup, DispatchSemaphore) 동시에 실행되는 작업들을 관리하는 방법에 대해서 알아보겠습니다. 여러 작업을 하나의 그룹으로 관리하고 하나의 리소스에 대한 접근을 통제해야 하는 문제들을 GCD를 활용하여 해결해보겠습니다. DispatchGroup DispatchGroup 은 디스패치 큐에 추가된 작업을 가상의 그룹으로 관리합니다. 서로 다른 디스패치 큐에 추가된 작업을 동일한 그룹에 추가하는 것도 가능합니다. 여러 작업을 하나의 작업으로 묶는 것이라고 생각하면 편합니다. 그러므로 그룹에 포함된 모든 작업이 완료되어야 그룹이 완료됩니다. 예시를 통하여 그룹이 어떻게 사용되는지 알아보겠습니다. let queue1 = DispatchQueue(label: "ConcurrentQueue", attributes: .concurrent) let ..