[펌] [문씨의 강좌] 메모리 관리2 <Property>
원문 : http://cafe.naver.com/mcbugi/71571
출처 : 맥부기 아이폰 카페
|
원문 : http://cafe.naver.com/mcbugi/71571
출처 : 맥부기 아이폰 카페
원문 : http://cafe.naver.com/mcbugi/71504
출처 : 맥부기 아이폰 카페
안녕하세요 문씨입니다
이번 강좌에서는 제일 중요한 기초인 메모리 관리에 대해 다루어 보겠습니다.
본론에 들어가기 전에 Objective-C에 대해 잠시 알아보겠습니다.
*이 부분은 제 개인적인 소견이므로 잘못된 정보가 있을 수도 있습니다.
Objective-C는 C++과 마찬가지로 C에서 파생된 언어입니다.
다만 구조적 특징을 보면 C++보다 Objective-C가 좀더 C에 가깝습니다.
C++은 C에서 메모리 관리를 직접 하는 포인터 개념에 충실한 반면 Objective-C는 메모리 관리를 어느정도 대신 해주는 개념을 가지고 있습니다.
저는 Objective-C가 가장 편합니다. (그동안 많이 써서 그렇겠지만;;;) 그리고 모바일의 제한적 성능 환경에 본의 아니게 가장 어울리는 언어라고 보고있습니다.
언어를 대체적으로 C나 C++, Java, Objective-C로 세분류로 나누고 본다면
일단 C나 C++은 개발하기가 어렵습니다. 하지만 메모리 관리를 직접할수 있기 때문에 고수가 개발한다면 정말 강력한 앱을 만들수도 있습니다.
Java는 메모리 관리를 신경 안쓰는 언어입니다. 덕분에 개발은 편하죠. 다만 메모리 관리하는 가비지콜렉션 때문에 모바일 환경에서는 상당히 느립니다.
*가비지콜렉션은 시스템상의 메모리를 전부 읽어서 안쓰는 메모리는 제거해 주는 개념입니다.
데스크탑이라면 별 의미없지만 모바일이라는 환경에서는 차이가 납니다.
그래서 직접 메모리를 관리 하는 방법 VS 가비지콜렉션으로 메모리 관리 라는 두 구도가 생성됩니다만 어느쪽이든 사용하기는 힘든 감이 있죠.
직접 메모리 관리는 너무 개발하기 힘들고 Java는 느려서 원하는 성능을 내기가 힘듭니다.
후에 모바일 기기의 성능이 PC급으로 올라가면 얘기가 달라지겠지만요.
그렇다면 개발하기는 편하면서 가비지컬렉션을 사용하지 않는 방법은 없을까 하는 질문에 등장하는것이 Objective-C입니다.
물론 Objective-C도 가비지콜렉션은 지원합니다. 하지만 모바일 기기인 아이폰이나 터치, 아이패드 등에서는 지원하지 않습니다. 즉 OSX에서는 지원하지만 iOS에서는 지원하지 않습니다.
애플도 가비지콜렉션의 무거움을 알기때문에 일부러 지원을 안했다고 합니다.
Objective-C에서는 리테인 카운트(retain count)기법을 사용합니다. 해당 객체를 몇군데에서 사용하고 있는지 그것을 수치로 가지고 있는겁니다. 그리고 아무곳에서도 사용하지 않을때 지워지는 개념입니다.
개념만 본다면 더이상 사용하지 않는 객체를 찾아서 지우는 가비지콜렉션과 유사하기도 합니다. 다른점이라면 가비지콜렉션은 시스템에 기생해서 메모리는 체크하는 방식이지만 Objective-C는 로직으로 제거되는 방식입니다.
즉 Java에서는 객체를 쓰고 뒤처리를 안하지만 Objective-C에서는 반드시 뒤처리를 합니다.
객체를 사용후 해당 객체에게 이곳에서는 사용을 중지했다고 알려주는겁니다.
Objective-C에서 모든 객체들의 순환(Life Cycle)을 이해 하려면 우선 NSObject를 알아야 합니다.
모든 객체는 NSObject를 상속 받았기 때문이죠. 그래서 NSObject를 알면 메모리관리를 알게 됩니다.
NSObject의 NS는 Next Step의 약자이고 Object는 말 그대로 객체를 뜻합니다.
먼저 NSObject의 헤더를 보겠습니다.
클래스 선언 부분인 @interface NSObject를 보면 최상위의 클래스 이기때문에 상속받는 부모클래스가 없습니다. 보통 @interface test : NSObject 식으로 부모클래스 선언이 반드시 있습니다. 물론 없으면 에러 경고를 띄우기도 합니다.
그리고 선언된 함수들을 보면 +와 -가 있습니다. +의 경우 클래스 함수(Class Method)라 하고 -는 인스턴트 함수(Instance Method)라 합니다. 클래스를 도장이라고 본다면 인스턴트는 도장으로 찍어서 만든 것이라 할수 있습니다. 클래스도 주소가 존재하기 때문에 객체로 취급 받기도 합니다. 물론 클래스는 단일 존재라 단 하나만 존재합니다. 그리고 인스턴트는 이 클래스로 원하는 만큼 만들어 낼수가 있습니다,
그럼 예를 보겠습니다.
NSObject *object = [NSObject alloc];
object라는 변수에 NSObject라는 클래스에 alloc이라는 클래스 함수를 호출해서 새로운 인스턴트 객체를 만든후 그 객체의 주소를 object변수에 저장했습니다.
그럼 다른 예를 보겠습니다.
NSArray *array = [NSArray alloc];
array라는 객체변수에 NSArray라는 클래스에 alloc이라는 클래스 함수를 호출해서 새로운 NSArray인스턴트 객체를 만들고 그 주소를 array에 지정합니다.
객체 생성은 반드시 alloc이라는 클래스 함수를 호출해서 원하는 객체를 찍어내는 겁니다.
모든 클래스는 NSObject를 상속받고 있기 때문에 alloc함수는 어디는 존재하게 되는겁니다.
객체를 생성했으면 이번에는 초기화를 해줍니다.
NSArray *array = [NSArray alloc];
[array init];
위에 헤더에도 있는 init은 기본 초기화 함수입니다. init은 인스턴트 함수이기 때문에 객체 생성후 최초로 객체에 호출되는 함수입니다.
단 위의 코드는 문제가 있습니다. 애플의 가이드 문서도 위와 같은 사용은 금지하고 있습니다.
왜그럴까요?
NSArray *array = [NSArray alloc]; 에서 객체를 생성한것까진 좋습니다.
문제는 [array init]; 입니다.
위 코드는 array에 init이라는 메세지만 날려줄뿐입니다.
init을 날렸는데 객체가 판단하기에 뭔가 잘못된 것이 있으면 객체가 스스로 해제 되기도 합니다.
위의 코드는 예문이라 그럴일은 절대 없지만 다른 코드에서는 그럴일도 있을수 있습니다.
init함수는 초기화가 제대로 되면 자기자신의 주소를 돌려주고 잘못되면 스스로 해제를 한 후 nil(0값)을 돌려줍니다.
즉 위의 코드는 init만 날리고 돌려주는 값에 대한 대처가 안되어 있기 때문에 array변수가 가지고 있는 객체의 주소는 쓰레기가 되어버립니다.
즉
NSArray *array = [NSArray alloc];
array = [array init];
아니면
NSArray *array = [[NSArray alloc] init];
을 사용합니다
결국 코드의 간결화를 위해 후자를 사용하게 됩니다.
이제 초기화를 했으면 해제를 해야합니다.
NSObject헤더에 인스턴트 함수중에 release와 autorelease가 있습니다.
autorelease는 잠시 후에 다루고 일단 release부터 봅니다.
만든 객체에 release를 날려주면 객체는 해제가 됩니다.
[array release];
단 여기서 많은 분들이 해매는 개념이 있습니다. 저 또한 이해하기 전까진 많이 해맸습니다.
release는 객체를 완전히 제거하는 명령이 아닌 단순히 이곳에서는 이 객체를 더이상 쓰지 않는다고 알려주는 함수입니다.
NSObject에는 객체를 생성하는 함수는 있지만 완전 제거하는 함수는 존재하지 않습니다.
C에서 본다면 free문이 없다는 것입니다.
대신 NSObject에는 retainCount라는 개념이 있습니다.
alloc으로 처음 생성될때 1이라는 수치로 시작됩니다.
그리고 release라는 함수가 호출되면 내부에 retainCount를 하나 줄입니다.
위와 같은 경우는 1에서 하나 줄어서 0이 됩니다.
그러면 객체는 자동으로 메모리 상에서 제거가 됩니다.
NSArray *array = [[NSArray alloc] init]; //1
[array release]; //0
Objective-C에서 메모리 누수란 이 retain count가 0이 안되고 남아있어서 벌어지는 문제입니다.
그러면 release가 있으니 반대도 있습니다.
retain이라는 함수가 그 반대 역활을 합니다.
NSArray *array = [[NSArray alloc] init]; // 1
[array retain]; //2
[array release]; //1
[array release]; //0
리테인 카운터의 개념 및 규칙은 “올린만큼 내려라” 입니다.
간단하죠?
근데 머리로는 이해 하겠는데 정작 코딩으로 하려고 하면 다시 해맵니다.
그리고 또 autorelease는 뭐고요.
autorelease의 존재는 정말 코딩을 편하게 해주는 기능입니다.
지금 이대로 autorelease없이 코딩한다면 위와 같이 전부 객체 생성을 한후 사용이 끝난 부분에서 일일이 전부 release를 날려줘야 합니다.
코딩 하다보면 사용하는 변수가 점점 많아지는데 이것들은 언제 일일이 release하나요.
오토릴리즈(autorelease)를 사용안한다면
NSArray *array = [[NSArray alloc] init];
... 한참 사용후
[array release];
이런식으로 사용하겠지만
오토릴리즈를 사용한다면
NSArray *array = [[[NSArray alloc] init] autorelease];
...사용
그리고 사용후에는 release를 신경 쓸 필요가 없습니다.
왠지 일일이 뒤처리 할필요가 없어진듯합니다.
마치 Java같군요.
오토릴리즈라 한다면 언젠간 자동으로 해제된다는 것인데
어떤 원리 일까요?
오토릴리즈는 편한 존재이지만 그 원리는 이해 못하기 때문에 오히려 처음 배우는 사람에게는 혼란을 일으킵니다.
언제 해제될까? 라는 궁금증이 이 함수에 대한 신뢰를 잃게 되는것이죠.
그래서 불편하더라도 확실히 생성하고 확실히 릴리즈를 날리는 코더를 많이 볼수 있었습니다.
오토릴리즈는 오토릴리즈 풀(NSAutoreleasePool)과 연결되어 있습니다.
생성된 객체에 autorelease를 날려주면 해당 객체는 자동으로 오토릴리즈 풀에 등록됩니다.
해제를 시켜주는 것이 아닌 등록입니다. 다른 말로 수영장(pool)에 빠트려 놓는것이죠
그리고 특정 시간이 지나면 그 수영장은 물을 빼고 청소를 합니다.
그때 전에 빠진것들은 다 릴리즈 처리가 되면서 해제되는 것이죠
그럼 NSAutoreleasePool이 모아서 한꺼번에 release처리해준다는 것은 알았습니다.
그러면 언제 그런일(drain)이 일어날까요?
저는 일단 프로세스가 이벤트 루프로 돌아갈때 일어난다고 보고있습니다.
앱이 실행중일때 아무것도 안하고 있다면
메인 스레드는 이벤트 루프에서 무한 루프를 돌고 있습니다.
그러다 어떠한 이벤트가 발생한다면 그 발생한 이벤트에 따라 함수를 호출합니다.
그리고 그 함수가 끝나면 다시 이벤트 루프로 돌아옵니다.
그리고 실행된 함수는 혼자서 일하고 끝나기도 하지만 작업을 위해 함수가 다른 함수를 부르는 일이 반드시 생깁니다.
예를 하나 들겠습니다.
이벤트 루프에서 돌고 있다가 유저가 화면에 터치 했습니다.
그러면 이벤트 루프는 UIApplication객체에 터치 이벤트가 있음을 알리는 메세지를 날립니다.
*보통 Objective-C에서 메세지를 날린다고 하면 해당 객체에 함수를 호출한다고 보면 됩니다.
그럼 앱은 최상위 화면인 UIWindow에 메세지를 보내고
UIWindow는 위에 올려진 뷰에 메세지를 보낼것이고
올려진 뷰는 올려진 뷰위에 올려진 뷰에 메세지를 보내서
가장 상단에 올려진 뷰까지 메세지가 갔다올것입니다.
즉 함수가 함수를 호출하는 상황입니다.
그래서 그사이에 생성되는 오토릴리즈된 객체들은 해제되지 않고 있다가
일을 마치고 이벤트 루프로 돌아오면 전부 해제 처리되는겁니다.
결국 함수내에서 autorelease시킨 객체들이 함수 진행도중 해제 될까봐 걱정할 일은 없습니다.
그럼 하나의 함수내에서 만들어 쓰고 버리는 객체들은 release를 사용할 일이 없겠군요.
전부 오토릴리즈로 처리하면 되니까요.
그렇습니다. 근데 더 편하게 쓰게 도와주는것들이 있습니다.
NextStep시절부터 개발되어 오던 모든 클래스들은 한가지 규칙을 지키고 있습니다.
“객체를 돌려주는 함수는 반드시 오토릴리즈된 객체를 돌려준다”는 것입니다.
위에서 언급했던 클래스 함수들.
한가지 예를 들어보죠.
NSArray *array = [[[NSArray alloc] init] autorelease];
저처럼 [[[ <- 이거에 익숙해진 분들이라면 별 문제 없겠지만 이것들이 많아진다면
그것도 나름데로 스트레스입니다.
NSArray같은 경우는 array라는 클래스 함수를 제공합니다.
함수로 돌려주는 객체는 반드시 오토릴리즈 된 객체이다 라는 규칙을 따르기 때문에
NSArray *array = [NSArray array];
이 코드는 위의 alloc init autorelease를 사용한 코드와 같은 의미 입니다.
이렇게 기본적으로 Foundation에 내장되어 있는 NS계열의 객체들이나 아이폰부터 추가된 UIKit등 모든 클래스들이 alloc이 아닌 특별히 제공하는 클래스 함수들은 전부 오토릴리즈 된 객체를 생성합니다.
제가 몇번 봐드린 코드 중이 이런 코드도 있었습니다.
1: NSString *str = [[NSString alloc] initWithString:@”test”];
2: str = [str uppercaseString];
3: [str release];
이러면 에러가 나고 튕긴다는 문제였습니다.
1에서 오토릴리즈 안된 객체를 생성합니다.
2에서 str에 기존의 문자열을 대문자로 바꾼 새로운 개체를 돌려주는 함수를 실행해서 돌려준 새로운 오토릴리즈된 객체의 주소를 다시 str에 지정합니다.
이시점에서 기존의 str객체의 주소를 잃어버리게 되는겁니다. 당연 메모리 누수가 납니다.
그리고 str는 전부 대문자로 변환된 새로운 객체를 가지고 있게 됩니다
하지만 오토릴리즈된 객체라 3에서 release를 날리면 당연 문제가 됩니다.
여기서 놓친 부분은 uppercaseString함수가 새로운 오토릴리즈된 객체를 만들어서 돌려준다는 점입니다.
그럼 alloc의외에 클래스 생성 함수에 의해 만들어지는 객체들은 전부 오토리리즈된 객체란것은 알았습니다.
그런데 아직도 이해가 안되는 부분이 있습니다.
NSArray같은 데이터 관리 객체는 어떻게 하나요?
NSArray보다 NSMutableArray를 예로 들겠습니다
NSMutableArray에는 객체를 추가하는 함수가 있습니다
addObject:인데
하나 예를 들죠
NSString *str = @”test”;
일단 문자열 객체를 준비합니다
NSMutableArray *array = [NSMutableArray array];
더이상 alloc init을 볼일이 없이 바로 오토릴리즈 된 배열 객체를 준비합니다.
[array addObject:str];
array 객체에 str을 넣었습니다.
간단히 생각하면 넣었다 라고 보겠지만
실제로 메모리 관리는 어떻게 될까요?
NSMutableArray는 객체를 새로 추가 받는 함수가 실행되면 자동으로 그 객체에게 retain 함수를 날립니다.
즉 여기서는 더이상 str을 쓸일이 없더라도 array는 str을 받은 이상 계속 가지고 있어야할 책임이 생긴겁니다.
리테인 카운트를 올려두었기 때문에 현재 코드에서 더이상 안쓴다 해도 객체는 유지됩니다.
하지만 별로 신경쓸 문제는 아닙니다.
NSMutableArray클래스는 객체를 받으면 책임 지기때문에 해제또한 책임 집니다.
즉 array가 오토 릴리즈 상태이기때문에 나중에 해제 될때
가지고 있던 모든 객체에게 다 release를 날려주게 됩니다.
근데 넣어둔 객체 메모리 관리를 한답시고 직접 release를 날려버리면
해당 str객체는 해제 되겠지만 array는 객체가 있는줄 알고
쓰레기 주소에 메세지를 때렸다가 베드 엑세스 에러가 나고맙니다.
결국 자기가 받은 객체는 자신이 책임 지는 구조입니다.
그렇다면 클래스에 선언한 전역 변수는 어떻게 될까요?
함수 내부에서 만들어 쓰고 끝나는 것들은 전부 오토릴리즈로 사용하겠지만 전역변수는 다릅니다. 더이상 안쓸때까지 retainCount를 하나 올려둬야 합니다.
그리고 객체가 해제될때 호출되는 dealloc함수에서 해당 전역 변수의 객체들을 다 해제 시켜줍니다.
처음 생성할때 alloc init으로 가지고 있어도 되고, 아니면 다른 함수를 통해 오토릴리즈된 객체를 가지고 있다면 이럴때 retain을 사용하면 됩니다.
title = [[NSString alloc] initWithString:@”tes”];
OR
title = [[커스텀클래스 getTitle] retain];
그리고
- (void)dealloc {
[title release];
...
즉 객체가 해제될때 가지고 있던 객체들도 같이 해제 해주는겁니다.
그럼 전역 변수도 알았습니다.
프로퍼티는 무엇일까요?
여기까지 이해했다 하더라도
object.title = ?
식의 코드를 보면 또다시 해맵니다.
1. object.title = [[NSString alloc] initWithString:@”Test”];
2. object.title = @”Test”;
3. object.title = [NSString stringWithString:@”Test”];
위의 1,2,3중 어느것이 맞다고 봅니까?
2, 3번이 맞는겁니다. 보통 문자열의 경우 2번이 제일 간단하면 3번과 별차이 없기때문에 2번을 사용합니다.
위와같은 점문법은 보통 프로퍼티(속성)로 정의되어 있는 경우입니다.
프로퍼티는 C++배울때 많이 듣던 setter, getter입니다.
즉 외부에서 해당 객체의 내부 변수에 접근할수 있도록 접근용 함수를 만드는 것입니다.
프로퍼티 선언은 먼저 헤더에서 합니다.
그리고 본문에서 @synthesize 선언을 해주면 정의한 속성이 설정됩니다.
그리고 위 선언을 하면 컴파일러에서 자동으로
- (NSString *)title이라는 getter함수와
- (void)setTitle:(NSString)이라는 setter함수가 자동 생성됩니다.
그리고 프로퍼티 설정에서 주의해야 할 부분이 있습니다.
retain이라는 부분인데 이부분이 retain, assign, 그리고 copy일수 있습니다.
getter함수는 단순히 그 변수를 돌려주면 되기때문에 특별한 조치는 없고
setter함수가 위 세가지 설정에 따라 내용이 달라집니다.
먼저 retain
retain이 설정되어 있으면 기본적으로 setter에서 기존의 title에 뭔가 있을 경우를 대비에 release를 먼저 날려주고 새로운 객체를 retain걸어주면서 받습니다.
assign의 경우
보통 델리게이트 등 객체의 주소만 지정할때 사용됩니다.
마지막으로 copy 는 retain과 비슷하지만 retain부분 대신 copy가 들어가서 객체를 복사할때 사용됩니다.
객체.title라고 써서 읽어 오는것은 [객체 title]로 써있는것과 같은 역활을 하고
객체.title = ? 식으로 지정하는것은 [객체 setTitle:?]식으로 설정 함수를 호출하는 것과 같습니다.
setter의 경우 앞서 다루어 봤던 NSMutableArray에서 addObject와 비슷하죠?
즉 속성으로 호출되는 것들은 다 함수입니다.
변수라고 착각을 하기때문에 위에서 1번과 같은 경우가 생기는 겁니다.
프로퍼티가 retain이나 copy인 경우 dealloc에서
어느쪽이 맞다고 생각합니까?
둘다 틀렸습니다.
1번은 전역인데 release안하고 넘어가서 문제가 되고
2번은 title함수로 불러온 title객체에 release를 때려서 객체는 해제 되었지만
title객체변수는 여전히 주소를 쥐고 있어서 문제가 될 여지가 있습니다
맞는 방법은
[title release];
title = nil;
아니면
self.title = nil;
입니다
후자의 경우
[self setTitle:nil]; 이 실행된 경우인데
위 retain 경우 함수를 보면
먼저 기존에 가지고 있던 객체에는 release를 보냅니다
문제없이 해제되고 그다음
title = [newTitle retain];에서 newTitle이 nil값이기때문에
title = [nil retain];이 됩니다
재미있는 점은 nil에는 무슨 메세지를 날려도 아무런 문제가 없습니다
0 x 어떤 숫자든 = 0인 원리
즉 title = nil이 됩니다
자 그럼 정리해 보겠습니다.
특의한 경우로 NSTimer같은 경우는 오토릴리즈 되어 있어도 타이머가 돌고있는 상태에는 시간이 지나도 해제를 안합니다 타이머가 정지되면 동시에 해제되어 버립니다.
리테인 했었으면 릴리즈 해주고 안했었으면 그냥 nil처리 해주면 무난합니다.
마치며...
Objective-C는 이름 그대로 객체 지향에 특화된 언어입니다.
객체간의 서로가 왠만해선 참견 안하는것이 규칙으로 되어 있습니다.
필요한 호출만 서로 해주고 내부는 어떻게 돌아가는지 신경안쓰는 것이죠.
차를 그저 운전할뿐이지 엔진 설계도 까지 읽어보나요?