C++ Aggregate Initialization | "집합체 초기화" 가이드

C++ Aggregate Initialization | "집합체 초기화" 가이드

이 글의 핵심

집합체는 조건을 만족하는 구조체·배열로, 중괄호로 멤버를 직접 채웁니다. C++20 지정 초기화, 기본·값·리스트 초기화와의 차이, std::string 등 비 POD 멤버가 있을 때의 주의점을 함께 보면 좋습니다.

집합체 초기화란?

집합체(aggregate) 는 사용자 정의 생성자·비공개 멤버·가상 등이 없는 구조체·배열이며, 중괄호 {}로 한 번에 초기화할 수 있습니다. 리스트 초기화, 값 초기화, 지정 초기화와 함께 보면 초기화 규칙을 정리하기 좋습니다.

struct Point {
    int x;
    int y;
};

Point p = {10, 20};  // 집합체 초기화

왜 필요한가?:

  • 간결성: 생성자 없이 멤버를 직접 초기화
  • 안전성: 모든 멤버를 명시적으로 초기화
  • 가독성: 구조체 정의와 초기화가 명확
// ❌ 생성자 방식: 코드가 길어짐
struct Point {
    int x, y;
    Point(int x_, int y_) : x(x_), y(y_) {}
};
Point p(10, 20);

// ✅ 집합체 초기화: 간결
struct Point {
    int x, y;
};
Point p = {10, 20};

집합체 조건

// ✅ 집합체
struct Aggregate {
    int x;
    double y;
};

// ❌ 비집합체
struct NonAggregate {
    int x;
private:
    int y;  // 비공개 멤버
};

집합체 조건 상세:

C++17/C++20 기준으로 다음 조건을 모두 만족해야 집합체입니다:

  1. 배열 또는 클래스 타입 (구조체, 클래스, 공용체)
  2. 사용자 정의 생성자 없음 (컴파일러 생성 생성자는 OK)
  3. 비공개/보호 멤버 없음 (모든 멤버가 public)
  4. 가상 함수 없음
  5. 가상/비공개/보호 기반 클래스 없음 (C++17부터 public 상속은 OK)
// ✅ 집합체 예시
struct Aggregate1 {
    int x;
    double y;
};

struct Aggregate2 : Aggregate1 {  // C++17: public 상속 OK
    int z;
};

// ❌ 비집합체 예시
struct NonAggregate1 {
    int x;
    NonAggregate1() = default;  // 사용자 정의 생성자
};

struct NonAggregate2 {
    int x;
private:
    int y;  // 비공개 멤버
};

struct NonAggregate3 {
    virtual void func() {}  // 가상 함수
};

집합체 판별 방법:

#include <type_traits>

struct Point { int x, y; };

static_assert(std::is_aggregate_v<Point>);  // true

// 사용 예시
template<typename T>
void initIfAggregate(T& obj) {
    if constexpr (std::is_aggregate_v<T>) {
        obj = {};  // 집합체 초기화
    }
}

실전 예시

예시 1: 구조체 초기화

struct Person {
    std::string name;
    int age;
    double height;
};

// 집합체 초기화
Person p1 = {"Alice", 30, 165.5};
Person p2{"Bob", 25, 175.0};

// 일부 생략
Person p3 = {"Charlie", 35};  // height는 0.0
Person p4 = {"David"};         // age는 0, height는 0.0

예시 2: 배열 초기화

int arr1[5] = {1, 2, 3, 4, 5};
int arr2[5] = {1, 2};  // 나머지는 0

// 2차원 배열
int matrix[2][3] = {
    {1, 2, 3},
    {4, 5, 6}
};

예시 3: 중첩 구조체

struct Address {
    std::string city;
    int zipCode;
};

struct Person {
    std::string name;
    Address address;
};

Person p = {
    "Alice",
    {"Seoul", 12345}
};

std::cout << p.name << std::endl;
std::cout << p.address.city << std::endl;

예시 4: C++20 지정 초기화

struct Config {
    int width = 800;
    int height = 600;
    bool fullscreen = false;
};

// C++20: 지정 초기화
Config cfg = {
    .width = 1920,
    .height = 1080,
    .fullscreen = true
};

다른 초기화 방식과 비교 (집합체 관점)

집합체 초기화는 보통 중괄호 리스트로 이루어지며, 리스트 초기화·값 초기화와 겹칩니다.

문법의미 (집합체 S)
S a;기본 초기화 — 지역이면 멤버가 쓰레기일 수 있음
S a{};값 초기화 — 멤버를 0·빈 값으로 맞추는 쪽(타입에 따라 0 초기화 포함)
S a = {1, 2};집합체 초기화(복사 리스트 형태)
S a{1, 2};집합체 + 리스트 초기화

비집합체(사용자 정의 생성자가 있는 클래스)는 S a{...}생성자 호출로 가므로, “집합체 초기화”라는 용어는 엄격히는 집합체 타입에만 해당합니다.

구조체 초기화 실전 패턴

  • 한 줄에 의미 부여: API 응답·설정값을 struct로 묶고 return { ok, msg };처럼 쓰면 필드 이름이 문서 역할을 합니다.
  • 기본 멤버 초기화 + {}: C++11 이후 멤버에 = 0 또는 = {}를 두고, 전체는 Config c{}로 통일하면 “어떤 필드도 빠뜨리지 않았는가”를 코드 리뷰하기 쉽습니다.
  • 중첩: 바깥 구조체 초기화 시 안쪽은 또 다른 집합체로 {...}를 한 단계 더 두면 됩니다(이미 예시 3).

C++20 지정 초기화(designated initializers) 심화

C++20에서는 C99 스타일의 .멤버 = 값 문법이 집합체 초기화에 들어옵니다. 선언 순서와 동일한 순서로만 써야 하며, 건너뛰면 생략된 멤버는 값 초기화됩니다.

struct Point { int x; int y; int z = 0; };

Point p = {.x = 1, .y = 2};       // z는 기본값 0 또는 값 초기화 규칙
// Point q = {.y = 2, .x = 1};   // 순서 어긋남 → C++20에서 ill-formed

장점

  • 필드 의미가 이름으로 드러나 큰 struct에서 실수(인자 순서 바꿈)가 줄어듭니다.
  • 리스트 초기화와 같이 쓰면 가독성이 좋아집니다.

주의

  • C와의 호환: C에서는 지정 초기화가 더 유연한 부분이 있었으나, C++20은 위에서 말한 순서 제약이 엄격합니다.
  • 비집합체에서는 집합 규칙 밖이므로, 생성자를 쓰는 타입은 생성자 인자로 설계하는 편이 맞습니다.

기본값

struct Data {
    int x = 10;      // 기본값
    int y = 20;
    int z = 30;
};

Data d1 = {};           // {10, 20, 30}
Data d2 = {100};        // {100, 20, 30}
Data d3 = {100, 200};   // {100, 200, 30}

기본값 동작 원리:

집합체 초기화 시 일부 멤버를 생략하면, 생략된 멤버는:

  1. 멤버 기본값이 있으면 그 값 사용
  2. 멤버 기본값이 없으면 값 초기화 (0, nullptr, false 등)
struct Config {
    int width = 800;    // 기본값 있음
    int height = 600;   // 기본값 있음
    bool fullscreen;    // 기본값 없음
};

Config c1 = {};                    // {800, 600, false} (모두 기본값)
Config c2 = {1920};                // {1920, 600, false} (width만 지정)
Config c3 = {1920, 1080};          // {1920, 1080, false}
Config c4 = {1920, 1080, true};    // {1920, 1080, true} (모두 지정)

실무 활용: 옵셔널 파라미터:

struct HttpRequest {
    std::string url;
    std::string method = "GET";  // 기본값
    int timeout = 30;            // 기본값
    bool followRedirects = true; // 기본값
};

// 필수 파라미터만 지정
HttpRequest req1 = {"https://example.com"};
// {url: "https://example.com", method: "GET", timeout: 30, followRedirects: true}

// 일부 파라미터 지정
HttpRequest req2 = {"https://example.com", "POST", 60};
// {url: "https://example.com", method: "POST", timeout: 60, followRedirects: true}

자주 발생하는 문제

문제 1: 순서

struct Point {
    int x;
    int y;
};

// ❌ 순서 바뀜
// Point p = {.y = 20, .x = 10};  // C++20에서 에러

// ✅ 순서 유지
Point p = {.x = 10, .y = 20};

문제 2: 생성자 있는 클래스

// ❌ 비집합체
struct NonAggregate {
    int x;
    NonAggregate(int v) : x(v) {}
};

// NonAggregate obj = {10};  // 에러

// ✅ 생성자 호출
NonAggregate obj(10);

문제 3: 상속

// C++17: 상속된 집합체도 가능
struct Base {
    int x;
};

struct Derived : Base {
    int y;
};

Derived d = {{10}, 20};  // Base{10}, y=20

문제 4: 배열 크기

// ✅ 크기 명시
int arr1[5] = {1, 2, 3};

// ✅ 크기 추론
int arr2[] = {1, 2, 3};  // 크기 3

// ❌ 초과
// int arr3[2] = {1, 2, 3};  // 에러

초기화 방식 비교

struct Point {
    int x, y;
};

// 집합체 초기화
Point p1 = {10, 20};
Point p2{10, 20};

// 생성자 호출 (생성자 있으면)
Point p3(10, 20);

// 기본 초기화
Point p4;  // x, y는 쓰레기 값

초기화 방법 비교표:

방법문법집합체비집합체특징
집합체 초기화Point p = {10, 20};멤버 직접 초기화
리스트 초기화Point p{10, 20};좁히기 방지
직접 초기화Point p(10, 20);생성자 호출
기본 초기화Point p;⚠️⚠️쓰레기 값 가능
값 초기화Point p{};0 또는 기본값

실무 권장:

// ✅ 집합체: 중괄호 초기화
struct Config {
    int width, height;
};
Config cfg = {800, 600};

// ✅ 비집합체: 생성자 호출
class Widget {
    int value_;
public:
    Widget(int v) : value_(v) {}
};
Widget w(10);

// ✅ 안전한 초기화: 값 초기화
Config cfg2{};  // {0, 0}

실무 패턴

패턴 1: 설정 구조체

struct ServerConfig {
    std::string host = "localhost";
    int port = 8080;
    int maxConnections = 100;
    bool enableLogging = true;
};

// 기본값 사용
ServerConfig cfg1 = {};

// 일부만 변경
ServerConfig cfg2 = {"0.0.0.0", 3000};

// C++20: 지정 초기화
ServerConfig cfg3 = {
    .host = "192.168.1.1",
    .port = 9000
};

패턴 2: 반환값 최적화

struct Result {
    bool success;
    std::string message;
    int errorCode;
};

Result processData(const std::string& data) {
    if (data.empty()) {
        return {false, "Empty data", 1};
    }
    
    // 처리 로직
    return {true, "Success", 0};
}

패턴 3: 테스트 데이터

struct TestCase {
    std::string input;
    std::string expected;
    bool shouldPass;
};

std::vector<TestCase> tests = {
    {"hello", "HELLO", true},
    {"world", "WORLD", true},
    {"", "", false}
};

흔한 실수와 함정 (추가)

  • std::string이 있는 집합체를 {0}만으로…
    멤버가 std::string이면 전부 0으로 두려면 각 멤버 규칙에 맞는 값 초기화가 필요합니다. S{}가 일반적으로 안전합니다. memset으로 0 채우기는 절대 하지 마세요.
  • 지정 초기화와 순서
    선언 순서와 다르게 쓰면 C++20에서 에러입니다. 리팩터링으로 멤버 순서를 바꾸면 지정 초기화 목록도 함께 고쳐야 합니다.
  • 집합체가 아니게 된 뒤
    virtual이나 사용자 정의 생성자를 추가하면 갑자기 T a = {1,2,3}깨지거나 생성자 경로로 바뀝니다. 공개 API의 struct는 변경 시 영향 범위를 넓게 봐야 합니다.

FAQ

Q1: 집합체 조건은 무엇인가요?

A:

  • 모든 멤버가 public
  • 사용자 정의 생성자 없음
  • 가상 함수 없음
  • C++17부터 public 상속 허용

Q2: 일부 멤버를 생략할 수 있나요?

A: 가능합니다. 생략된 멤버는 멤버 기본값 또는 값 초기화(0, nullptr 등)됩니다.

struct Point { int x, y, z = 100; };
Point p = {10};  // {10, 0, 100}

Q3: 초기화 순서는?

A: 선언 순서대로 초기화됩니다. C++20 지정 초기화도 선언 순서를 따라야 합니다.

Q4: C++20에서 어떤 변화가 있나요?

A: 지정 초기화(designated initializers) 지원이 추가되었습니다.

struct Point { int x, y; };
Point p = {.x = 10, .y = 20};  // C++20

Q5: 배열 크기는 어떻게 결정되나요?

A:

  • 명시적: int arr[5] = {1, 2, 3}; (크기 5)
  • 추론: int arr[] = {1, 2, 3}; (크기 3)

Q6: 중첩 구조체는 어떻게 초기화하나요?

A: 중괄호를 중첩하여 초기화합니다.

struct Inner { int a, b; };
struct Outer { Inner inner; int c; };

Outer o = {{10, 20}, 30};  // inner={10, 20}, c=30

Q7: 집합체 초기화 학습 리소스는?

A:

관련 글: 리스트 초기화, 값 초기화, 지정 초기화, 기본 초기화.

한 줄 요약: 집합체 초기화는 중괄호로 구조체나 배열의 멤버를 직접 초기화하는 간결한 방법입니다.


같이 보면 좋은 글 (내부 링크)

이 주제와 연결되는 다른 글입니다.

  • C++ List Initialization | “리스트 초기화” 가이드
  • C++ Value Initialization | “값 초기화” 가이드
  • C++ Designated Initializers | “지정 초기화” 가이드
  • C++ Default Initialization | “기본 초기화” 가이드

관련 글

  • C++ Aggregate Initialization 완벽 가이드 | 집합 초기화
  • C++20 Designated Initializers 완벽 가이드 | 명확한 구조체 초기화
  • C++ call_once |
  • C++ 배열 vs vector |
  • C++ struct vs class |