프로그래밍 언어/C++

C++ 중첩 클래스 (Nested Class)

ShovelingLife 2022. 7. 20. 19:53

중첩 클래스의 이름은 포함 클래스의 범위에 있으며 중첩 클래스의 멤버 함수에서 이름을 조회하면 중첩 클래스의 범위를 검사 한 후 포함 클래스의 범위를 방문한다. 둘러싸는 클래스의 다른 멤버와 마찬가지로 중첩 클래스는 둘러싸는 클래스가 액세스하는 모든 이름 (개인, 보호 등)에 액세스 할 수 있지만 그렇지 않으면 독립적이며 둘러싸는 클래스 this 포인터에 대한 특별한 액세스는 없다 .

중첩 클래스의 선언은 둘러싸는 클래스의 형식 이름, 정적 멤버 및 열거 자만 사용할 수 있다. (C ++ 11까지)
중첩 클래스의 선언 은 비 정적 멤버에 대한 일반적인 사용 규칙 에 따라 둘러싸는 클래스의 모든 멤버를 사용할 수 있다 . (C ++ 11부터)
int x,y; // globals

class enclose 
{ // 둘러싸는 클래스
    int x; // 참고 : 비공개 멤버
    static int s;
    
 public:
    struct inner { // 중첩 클래스
        void f(int i) {
            x = i; // 오류 : 인스턴스없이 비 정적 enclose :: x에 쓸 수 없다.
            int a = sizeof x; // C ++ 11까지 오류,
                              // C ++ 11에서 OK : sizeof의 피연산자는 평가되지 않는다.
                              // 비 정적 enclose :: x 사용이 허용된다.
            s = i;   // OK : 정적 enclose :: s에 할당 할 수 있다.
            ::x = i; // OK : 글로벌 x에 할당 가능
            y = i;   // OK : 전역 y에 할당 가능
        }
        void g(enclose* p, int i) 
        {
            p->x = i; // OK : enclose :: x 할당
        }
    };
};

중첩 클래스 내에 정의 된 친구 함수는 중첩 클래스 내에 정의 된 멤버 함수의 본문에서 조회가 둘러싸는 클래스의 개인 멤버를 찾을 수 있는 경우에도 클래스의 멤버에 대한 특별한 액세스 권한이 없다. 중첩 클래스의 멤버에 대한 클래스 외부 정의는 포함 클래스의 네임 스페이스에 나타난다.

struct enclose {
    struct inner {
        static int x;
        void f(int i);
    };
};
int enclose::inner::x = 1; // definition
void enclose::inner::f(int i) {} // definition

중첩 된 클래스는 동일한 선언 클래스 본문 내에서 또는 외부에서 앞으로 선언되어 나중에 정의 될 수 있다.

class enclose {
    class nested1; // 앞으로 선언
    class nested2; // 앞으로 선언
    class nested1 {}; // 중첩 된 클래스 정의
};
class enclose::nested2 { }; // 중첩 된 클래스 정의

중첩 클래스 선언은 멤버 액세스 지정자를 준수하며 , 해당 클래스의 오브젝트를 조작 할 수 있지만 개인 멤버 클래스는 클래스의 범위 밖에서 이름을 지정할 수 없다.

class enclose {
    struct nested { // 개인 회원
        void g() {}
    };
 public:
    static nested f() { return nested{}; }
};
 
int main()
{
    // enclose :: nested n1 = enclose :: f (); // 오류 : '중첩'은 비공개이다.
 
    enclose::f().g(); // OK : 'nested'라는 이름을 지정하지 않았다.
    auto n2 = enclose::f(); // OK : 'nested'라는 이름을 지정하지 않았다.
    n2.g();
}

 

출처 : https://runebook.dev/ko/docs/cpp/language/nested_classes