구조체(Structure)는 프로그래밍에 있어서 없어서는 안 되는 매우 중요한 요소입니다. 구조체는 '하나 이상의 변수를 묶어서 새로운 자료형을 정의하는 Custom data type'입니다. Go언어에서의 구조체는 필드들의 집합체이며 필드들의 컨테이너입니다. C언어와 같은 절차 지향 언어에서 구조체는 객체들의 공통되는 속성과 특징에 따라 쓰입니다. 예를 들어, 사람의 정보를 저장한다고 생각해봅시다. 사람의 정보로는 이름, 성별, 나이, 전화번호, 직업 등등이 있습니다. 그래서 사람을 추가할 때마다 각각의 정보를 선언하고 초기화 하는 것이 아니라 '사람'이라는 구조체 안에 묶어서 저장합니다. 즉, '정보 집합'이라는 것입니다.
하지만, Go언어는 객체 지향을 따른다고 했습니다. 여기서 주의해야 할 것은 Go언어는 전통적으로 객체 지향 언어의 클래스, 객체, 상속 개념이 없다는 것입니다. 그와 유사한 형태로 따르고 있는 것입니다. Go언어에서 클래스는 Custom 타입을 정의하는 구조체로 표현되는데, 이는 전통적인 객체지향의 클래스가 필드와 메소드를 함께 갖는 것과 달리 구조체는 필드만을 가지고, 메소드는 별도로 분리하여 정의하는 것입니다. 메소드와 관련된 구조체에 대한 설명은 '메소드' 강의에서 다루겠습니다.
'함수' 챕터에서 익명 함수를 배울 때 함수의 원형을 정의하기 위해 type문을 사용한 적이 있습니다. 구조체도 이와 마찬가지로 Custom data type이기 때문에 type문을 사용해서 구조체를 정의합니다. 아래에 구조체의 선언 형태를 알아보기 위해 사람의 정보를 만들 수 있는 구조체를 만들었습니다.
그리고 구조체 객체를 생성하는 방법에 대해서 알아보겠습니다. 구조체 선언은 단순히 형태의 선언입니다. "앞으로 person이라는 형은 이렇게 쓸거야."라고 한 것일 뿐, 그 형을 이용해 아무것도 만들지 않으면 아무 의미가 없습니다. 따라서 구조체는 선언 후 객체를 생성해서 사용할 수 있습니다.
구조체 객체 선언은 지난 강의에서 배운 '컬렉션 - 슬라이스, 맵'과 선언 방식이 유사합니다. 우선 특징에 대해 알아보겠습니다.
구조체 객체를 생성하려면 "객체이름 := 구조체이름{저장할값}"으로 입력해 선언과 동시에 초기화를 할 수 있습니다. 만약에 저장할값에 아무것도 입력하지 않는다면 빈 객체가 생성됩니다. 만약 빈 객체로 생성했다면 "객체이름.필드이름 = 저장할값"으로 저장할 수 있습니다. '.(dot)'을 이용하면 필드 값을 저장하는 기능만 있는 것이 아니라 필드 값을 접근할 수 있는 용법입니다. 그리고 빈 객체 혹은 만약 일부 필드가 생략될 경우 생략된 필드들은 Zero value (정수인 경우 0, float인 경우 0.0, string인 경우 "", 포인터인 경우 nil 등)를 갖습니다. 이 용법에 대해 아래 간단한 코드로 확인해보겠습니다.
아래 코드를 바로 실행해보세요.
위 코드에서 눈여겨 봐야 할 것은 선언과 동시에 초기화 할 때 두 가지 방법이 있다는 것입니다. 초기화 할 때 값을 나열하면 구조체에 선언한 필드 순서대로 저장됩니다. 하지만 필드 이름에 값을 지정한다면 순서에 상관없이 해당 필드에 값이 저장됩니다. 그리고 Go언어의 구조체는 기본적으로 'mutable' 개체로서 필드 값이 변화할 경우 별도로 새 개체를 만들지 않고 해당 개체 메모리에서 직접 변경됩니다. 위 코드에서 p3.name = "ryu"
를 입력해 값을 직접 수정했습니다.
그리고 함수(메소드)에서 같이 값을 복사해서 지역 변수로 사용하는 경우가 아니라 원래 값의 주소를 참조해 값이 저장된 주소에 직접 접근 하는 경우에 포인터를 썼습니다. 매개변수에 '&'을 붙여서 Pass by reference를 한 것입니다. 구조체도 마찬가지로 '구조체 포인터'를 생성할 수 있습니다. 구조체 포인터를 생성하는 방법은 두 가지가 있습니다.
- 'new(구조체이름)'을 사용하여 객체를 생성하기.
- 구조체 이름 앞에 & 붙이기.
주의할 점은, 다른 자료형의 포인터들은 역참조를 위해 '*' 연산자를 사용했습니다. 하지만 포인터 구조체는 선언하면 자동으로 역참조 됩니다. 따라서 함수 안에서 * 연산자를 사용할 필요가 없습니다.
아래 구조체 포인터를 만드는 간단한 예시가 있습니다. 바로 실행해보세요.
생성자(constructor) 함수
구조체는 사용자 임의로 하나 이상의 변수를 묶어 새로운 자료형을 정의한 것이라고 했습니다. 이때, 구조체를 사용하기 위해서는 우선 객체를 생성해야 사용할 수 있습니다. 그런데 때로는 구조체의 필드 자체가 사용 전에 초기화되어야 하는 경우가 있습니다. 예를 들어, 구조체의 필드가 'map' 형일 경우 구조체를 초기화할 때마다 맵 필드도 같이 초기화해야 하는 번거로움이 있을 수 있습니다. 따라서 사전에 미리 초기화를 해 놓으면 외부 구조체 사용자가 매번 맵을 초기화해야 한다는 것을 기억할 필요가 없습니다.
이러한 목적을 위해 '생성자 함수'를 사용할 수 있습니다. 생성자 함수는 호출하면 구조체 객체 생성 및 초기화, 입력한 필드 생성 및 초기화함과 동시에 구조체를 반환합니다.
위 생성자 함수는 구조체 객체를 포인터와 함께 반환합니다. 포인터 값이 없는 객체를 생성하는 생성자를 만들려면 반환형에 구조체 이름 앞에 붙은 포인터 연산자를 없애면 됩니다.
위 생성자 함수를 이용해 구조체를 만드는 아래 예시 코드를 실행해보세요.
위 예시에서 s1
객체는 생성자 함수로 data
필드의 맵을 초기화했기 때문에 바로 data
필드에 값을 저장할 수 있습니다. 하지만 s2
객체는 구조체만 생성했기 때문에 data
필드에 값을 저장하기 위해 선언이 필요한 맵은 따로 초기화해야 합니다.
이렇게 생성자 함수를 사용하면 구조체의 사용이 훨씬 수월해질 수 있습니다. 형태를 익히기 바랍니다.