C++学习之嵌套类和局部类

时间:2023-03-08 21:15:29
C++学习之嵌套类和局部类

局部类

在一个函数体内定义的类称为局部类。 局部类中只能使用它的外围作用域中的对象和函数进行联系,因为外围作用域中的变量与该局部类的对象无关。在定义局部类时需要注意:局部类中不能说明静态成员函数,并且所有成员函数都必须定义在类体内。在实践中,局部类是很少使用的。下面是一个局部类的例子。

int a;

void fun()

{

  static int s;

class A

{

  public:

  void init(int i) { s = i; }

};

A m;

  m.init(10);

}

嵌套类理解

在一个类中定义的类称为嵌套类,定义嵌套类的类称为外围类。定义嵌套类的目的在于隐藏类名,减少全局的标识符,从而限制用户能否使用该类建立对象。这样可以提高类的抽象能力,并且强调了两个类 (外围类和嵌套类) 之间的主从关系。

class A

{

public:

    class B 

{   

public:

      …

    private:

      …

    };

    void f();

private:

    int a;

}

其中,类B是一个嵌套类,类A是外围类,类B定义在类A的类体内。

对嵌套类的若干说明:

1、从作用域的角度看,嵌套类被隐藏在外围类之中,该类名只能在外围类中使用。如果在外围类的作用域内使用该类名时,需要加名字限定。

2、从访问权限的角度来看,嵌套类名与它的外围类的对象成员名具有相同的访问权限规则。不能访问嵌套类的对象中的私有成员函数,也不能对外围类的私有部分中的嵌套类建立对象。

3、嵌套类中的成员函数可以在它的类体外定义。

4、嵌套类中说明的成员不是外围类中对象的成员,反之亦然。嵌套类的成员函数对外围类的成员没有访问权,反之亦然。

在分析嵌套类与外围类的成员访问关系时,往往把嵌套类看作非嵌套类来处理。这样,上述的嵌套类可写成如下格式:

class A

{  public:

    void f();

  private:

    int a;

};

class B

{

 public:

   …

private:

  …

};  [ 嵌套类的理解形式 ]

 

5、在嵌套类中说明的友元对外围类的成员没有访问权。

6、如果嵌套类比较复杂,可以只在外围类中对嵌套类进行声明,关于嵌套的详细的内容可在外围类体外的文件域中进行定义。(理解这句话)

嵌套类规则详细分析

1、嵌套类的名字只在外围类可见。

在一个类的内部定义另一个类,我们称之为嵌套类(nested class),或者嵌套类型。之所以引入这样一个嵌套类,往往是因为外围类需要使用嵌套类对象作为底层实现,并且该嵌套类只用于外围类的实现,且同时可以对用户隐藏该底层实现。

       虽然嵌套类在外围类内部定义,但它是一个独立的类,基本上与外围类不相关。它的成员不属于外围类,同样,外围类的成员也不属于该嵌套类。嵌套类的出现只是 告诉外围类有一个这样的类型成员供外围类使用。并且,外围类对嵌套类成员的访问没有任何特权,嵌套类对外围类成员的访问也同样如此,它们都遵循普通类所具有的标号访问控制。

若不在嵌套类内部定义其成员,则其定义只能写到与外围类相同的作用域中,且要用外围类进行限定,不能把定义写在外围类中。例如,嵌套类的静态成员就是这样的一个例子。(如何完成?)

前面说过,之所以使用嵌套类的另一个原因是达到底层实现隐藏的目的。为了实现这种目的,我们需要在另一个头文件中定义该嵌套类,而只在外围类中前向声明这个嵌套类即可。当然,在外围类外面定义这个嵌套类时,应该使用外围类进行限定。使用时,只需要在外围类的实现文件中包含这个头文件即可。

另外,嵌套类可以直接引用外围类的静态成员、类型名和枚举成员,即使这些是private的。一个好的嵌套类设计:嵌套类应该设成私有。嵌套类的成员和方法可以设为public。

嵌套类名字的解析过程

嵌套类定义的名字解析过程:

出现在名字使用点前的嵌套类的声明。

出现在名字使用点前外围类的声明。

嵌套类定义前名字空间域的声明。

嵌套类的成员定义中的名字解析过程:

成员函数局部声明。

嵌套类成员的声明。

外围类成员的声明。

成员函数定义前名字空间域中出现的声明。

参考实例

#include <iostream>

using namespace std;

class A

{

private:

int n;

public:

A(int n)

{

this->n=n;

cout<<"Aconstructor"<<endl;

}

voidshow();

class B

{

public:

B( )

{

cout<<"B constructor"<<endl;

}

voiddisp()

{

cout<<"B disp"<<endl;

}

};

Bb;

};

voidA::show()

{

cout<<n<<endl;

}

void main()

{

A a(1);

a.show();

a.b.disp();

}

class Outter

{

// friend class Inner 加了这行也白搭

(理解为什么)正确写法应该是;

protected:

voidFoo(){}

private:

voidFoo2(){}

public:

voidFoo3(){}

private:

classInner

{

Inner(Outter*pFather)

{

pFather->Foo();// Error, compiler complain !!

pFather->Foo2();//Error, compiler complain !!

pFather->Foo3();//OK !

}

};

};

嵌套的结构(当然包括类)并不能自动获得访问private成员的权限,要获得的话,必须遵守特定的规则:首先声明(不是定义)一个嵌套结构,然后声明它是全局范围使用的一个friend,最后定义这个结构。结构的定义必须与friend声明分开,否则不会把它看做成员。

改成这样,就可以通过编译了

class Outter

{

protected:

voidFoo(){};

private:

voidFoo2(){};

public:

voidFoo3(){};

private:

class Inner; // 声明  (是否可以省略声明?不可以)

friend Inner; // 声明为友元

classInner

{

Inner(Outter*pFather)

{

pFather->Foo();

pFather->Foo2();

pFather->Foo3();

}

};

};