[c/objetive-c] 불투명 타입

불투명타입(Opaque Type)

코어 파운데이션 관련 문서를 보면 불투명타입(Opaque Type)이라는 말이 자주 나온다. 코어 파운데이션에서 객체처럼 사용되는 모든 타입들은 불투명타입(CFArrayRef, CFStringRef 등등)이다. "불투명"하다고 해서 감이 쉽게 오지 않는데, 간단히 풀어 쓰자면 "내부의 데이터 구조를 들여다볼 수 없는 타입"이라고 보면 된다.

코코아에서 객체를 만들 때 인스턴스 변수는 기본적으로 protected 나 private으로 만들어지기 때문에 이러한 인스턴스 변수는 객체가 이를 읽거나 쓸 수 있는 메소드(accessor)를 제공하지 않는 이상 객체 속에 어떤 변수들이 있는지 외부에서는 알 수 없다.

불투명 타입도 마찬가지로 객체 내부의 데이터 구조를 감추고 대신 이러한 객체를 다룰 수 있는 API 함수를 통해서만 객체를 조작하도록 하기 위해 사용된다. 즉 내부 정보를 외부에서 들여다 볼 수 없도록 은닉하고 API 함수로만 이를 사용하도록 하는 것이다. 외부에서 내부를 들여다 볼 수 없기 때문에 '불투명 타입'이라고 말한다.

코어 파운데이션은 코코아의 객체들에 대응되는 불투명 타입을 C로 구현한 프레임워크이다. C에서의 객체는 주로 구조체를 사용하여 만들게 되는데, 구조체는 멤버변수들에 대해 점(dot)연산자를 통해(구조체 포인터는 -> 연산자를 사용해서) 접근할 수 있기 때문에 불투명하지 않다.

따라서 이러한 불투명 타입을 만들기 위해서는 void 포인터를 통해서 해당 구조체 포인터를 강제로 캐스팅해서 사용한다. 그리고 API 함수에서는 이 void 포인터를 다시 원래 구조체 포인터로 강제 캐스팅해서 내부의 멤버에 접근하는 방식을 따르면 된다.

예를 들어 연결 리스트를 불투명 타입으로 만든다고 하면 대충 다음과 같은 식으로 타입을 정하는 것이다.

struct __node {
    int value;
} node;
struct __list {
    node* prev;
    node* next;
    unsigned int length;
} list;
typedef void* list_ref;

이제 리스트의 생성자 함수는 다음과 같은 꼴이 된다.

list_ref initList(unsigned int length) {
    list* newList = (list*)malloc(sizeof(list));
    newList->head = (node*)malloc(sizeof(node));
    newList->tail = (node*)malloc(sizeof(node));
    newList->head->next = tail;
    newList->head->prev = NULL;
    newList->tail->prev = newList->head;
    newList->tail->next = NULL;
    newList->length = 0;
    return (list_ref)newList;
}

생성자 함수를 통해 양방향 연결 리스트를 만들어서 그 포인터를 돌려받지만, 타입 자체는 void* 형이므로 그 내부의 구조를 확인할 수 없다. (사용하는 쪽에서 list 구조체의 내용을 알고 있지 않은 이상) 만약 이 리스트의 끝에 새로운 값을 추가하는 API 함수를 작성한다고 하면, 다시 리스트 포인터를 구조체 원형의 포인터로 캐스팅한 다음, 사용해야 한다.

void addValue(list_ref aList, int value) {
    list* cList = (list*)aList;
    node *newNode = (node*)malloc(sizeof(node));
    newNode->value = value;
    cList->tail->prev->next = newNode;
    newNode->prev = cList->tail->prev;
    newNode->next = cList->tail;
    cList->tail->prev = newNode;
    cList->length += 1;
}

언어 자체가 객체 지향적인 특성을 쉽게 지원할 수 있도록 디자인된 C++이라면 객체 내에 private 변수를 선언하는 것으로 객체 내부의 정보를 은닉할 수 있지만, C라고 해서 이러한 캡슐화를 구현할 수 없는 것은 아니다.

또한 이렇게 불투명 타입을 만들어 쓰는 경우에는 생성자 함수에서 메모리 할당 작업을 한 후 생성된 포인터만 돌려주게되므로, 객체를 다쓴 후에 해제를 위해서는 별도의 Close…, Delete… 와 같은 함수를 만들어서 메모리 누수 없이 객체를 해제할 수 있도록 해야 한다.

Read more

워드프레스에서 고스트로 이전

워드프레스에서 고스트로 이전

이 글을 쓰면서도 믿기 힘든 사실인데, 블로그라는 걸 처음 시작한지가 20년이 되었습니다. 이글루스에서 처음 시작했다가, SK컴즈가 인수한다고 발표함과 동시에 워드프레스로 플랫폼을 옮겼죠. 워드프레스오 옮긴 이후에는 호스팅 환경을 이리 저리 옮기긴 했지만 거의 18년 가까이 워드프레스를 사용해온 것 같습니다. 그 동안 워드프레스는 블로깅 툴에서 명실상부한 범용CMS로 발전했습니다. 사실 웬만한 홈페이지들은 이제

By sooop
띄어쓰기에 대한 생각

띄어쓰기에 대한 생각

업무 메일을 쓸 때 가장 많이 쓰는 말 중에 하나가 메일 말미에 ‘업무에 참고 부탁 드립니다.‘인데요, 어느 날부터 아웃룩에서 이 ‘부탁 드립니다’가 틀렸다고 맞춤법 지적을 하기 시작했습니다. 맞는 말은 ‘부탁드립니다’라고 붙여 쓰는 거라고. 사실 아래아한글 시절부터 이전의 MS워드까지, 워드프로세서들의 한국어 맞춤법 검사 실력은 거의 있으나 마나 한

By sooop

구글 포토에서 아이클라우드로 탈출한 후기

한 때 구글 포토가 백업 용량을 무제한으로 제공해 주겠다고해서, 구글 포토를 사용해서 사진을 백업해왔습니다. 물론 이 이야기의 결말은 저나 이 글을 읽고 있는 여러분이나 모두 알고 있습니다. 사실 AI에게 학습 시킬 이미지 데이터를 모으기 위한 것일 뿐이라거나 하는 이야기는 그 당시에도 있었습니다만, 에이 그래도 구글인데 용량은 넉넉하게 주겠지…하는 순진한

By sooop

Julia의 함수 사용팁

연산자의 함수적 표기 Julia의 연산자는 기본적으로 함수이며, 함수 호출 표기와 같은 방식으로 호출하는 것이 가능합니다. 또한 그 자체로 함수이기 때문에 filter(), map() 과 같이 함수를 인자로 받는 함수에도 연산자를 그대로 적용하는 것이 가능합니다. 특히 + 연산자는 sum() 함수와 같이 여러 인자를 받아 인자들의 합을 구할 수 있습니다. 2 + 3 # = 5 +(2,

By sooop