Бьерн Страуструп - Язык программирования С++. Главы 5-7 - Вложенные классы
ОГЛАВЛЕНИЕ
Страница 16 из 70
5.4.3 Вложенные классы
Описание класса может быть вложенным. Например:           class set {
              struct setmem {
                 int mem;
                 setmem* next;
                 setmem(int m, setmem* n) { mem=m; next=n; }
              };
              setmem* first;
           public:
              set() { first=0; }
              insert(int m) { first = new setmem(m,first); }
              // ...
           };Доступность вложенного класса ограничивается областью видимости лексически объемлющего класса:setmem m1(1,0); // ошибка: setmem не находитсяЕсли только описание вложенного класса не является совсем простым, то лучше описывать этот класс отдельно, поскольку вложенные описания могут стать очень запутанными:
// в глобальной области видимости
         class setmem {
         friend class set;  // доступно только для членов set
           int mem;
           setmem* next;
           setmem(int m, setmem* n) { mem=m; next=n; }
           // много других полезных членов
        };
        class set {
           setmem* first;
        public:
           set() { first=0; }
           insert(int m) { first = new setmem(m,first); }
           // ...
        };Полезное свойство вложенности - это сокращение числа глобальных имен, а недостаток его в том, что оно нарушает свободу использования вложенных типов (см. $$12.3). Имя класса-члена (вложенного класса) можно использовать вне описания объемлющего его класса так же, как имя любого другого члена:
         class X {
            struct M1 { int m; };
         public:
            struct M2 { int m; };
            M1 f(M2);
        };
        void f()
        {   M1 a;      // ошибка: имя `M1' вне области видимости
            M2 b;      // ошибка: имя `M1' вне области видимости
            X::M1 c;   // ошибка: X::M1 частный член
            X::M2 d;   // нормально
        }Отметим, что контроль доступа происходит и для имен вложенных классов. В функции-члене область видимости класса начинается после уточнения X:: и простирается до конца описания функции. Например:
M1 X::f(M2 a) // ошибка: имя `M1' вне области видимости
{ /* ... */ }
X::M1 X::f(M2 a) // нормально
{ /* ... */ }
X::M1 X::f(X::M2 a) // нормально, но третье уточнение X:: излишне
{ /* ... */ }
