2022年

2022年发布的文章
  • C++类成员的访问范围(C++ private、public、protected)

    在类的定义中,可以用 private、public 和 protected 三种关键字来指定成员可被访问的范围。

    private:用来指定私有成员。一个类的私有成员,不论是成员变量还是成员函数,都只能在该类的成员函数内部才能被访问。

    public:用来指定公有成员。一个类的公有成员在任何地方都可以被访问。

    protected:用来指定保护成员。这需要等介绍“继承”之后再解释。

    三种关键字出现的次数和先后次序都没有限制。成员变量的可访问范围由离它前面最近的那个访问范围说明符决定。

    如果某个成员前面没有访问范围说明符,则对 class 来说,该成员默认地被认为是私有成员;对 struct 来说,该成员默认地被认为是公有成员。例如:

    class A
    {
        int m, n;
    public:
        int a, b;
        int func1();
    private:
        int c, d;
        void func2();
    public:
        char e;
        int f;
        int func3();
    };

    在上面的类 A 中,成员 a、b 和 func1 是公有的,c、d 和 func2 是私有的,e、f 和 func3 又是公有的。m 和 n 没有指定可访问范围,则是私有的。如果把 class 换成 struct,那么 m 和 n 就是公有的。

    下面的程序可以说明公有成员和私有成员的区别。假设一个企业员工管理程序的一小部分代码如下:

    #include <iostream>
    #include <cstring>
    using namespace std;
    class CEmployee {
    private:
        char szName[30];  //名字
    public:
        int salary;  //工资
        void setName(char* name);
        void getName(char* name);
        void averageSalary(CEmployee el, CEmployee e2);
    };
    void CEmployee::setName(char* name) {
        strcpy(szName, name);  //ok
    }
    void CEmployee::getName(char* name) {
        strcpy(name, szName);  //ok
    }
    void CEmployee::averageSalary(CEmployee el, CEmployee e2)
    {
        salary = (el.salary + e2.salary) / 2;
    }
    int main()
    {
        CEmployee e;
        strcpy(e.szName, "Tom1234567889");  //编译出错,不能访问私有成员
        e.setName("Tom");  //ok
        e.salary = 5000;  //ok
        return 0;
    }

    在上面的程序中,szName 是私有成员,其他成员都是公有的。

    私有成员只能在成员函数内部访问,因此第 14 行和第 17 行没有问题,这两条语句都是在访问函数所作用的那个对象的 szName 私有成员。另外,类的成员函数内部可以访问任何同类对象的私有成员。

    所谓成员函数内部,指的就是成员函数的函数体内部。main 函数中的语句,如第 26 行,当然就不在 CEmployee 的成员函数内部,因此该行试图访问 e 这个对象的 szName 私有成员变量就会导致编译错误。

    而第 27 行虽然也不属于 CEmployee 类的成员函数内部,但其访问的是对象 e 的公有成员 setName,因此没有问题。同理,第 28 行也没有问题。

    在 CEmployee 类的成员函数外面,若要访问 CEmployee 对象的 szName 私有成员变量,不能直接访问,只能通过两个成员函数 setName 和 getName 间接进行访问。

    “隐藏”的作用

    设置私有成员的机制叫作“隐藏”。“隐藏”的一个目的就是强制对成员变量的访问一定要通过成员函数进行。这样做的好处是,如果以后修改了成员变量的类型等属性,只需要更改成员函数即可;否则,所有直接访问成员变量的语句都需要修改。

    以上面的企业员工管理程序为例,如果 szName 不是私有的,那么整个程序中可能会有很多类似于第 26 行

    strcpy(man1, szName, "Tom1234567889");

    这样的语句。假设需要将该程序移植到内存空间紧张的手持设备上,希望将 CEmployee 类的成员变量 szName 改为 char szName[5],以便节约空间,那么所有这样的语句都要找出来检查一番并修改,以防止数组越界。这显然很麻烦。

    如果将 szName 变为私有的,则除了 CEmployee 类的成员函数内部,其他地方不可能出现第 26 行那样对 szName 直接访问的语句,所有对 szName 的访问都是通过成员函数进行的。例如:

    e.setName("Tom12345678909887");

    就算 szName 的长度变短了,上面的语句也不需要修改,只要修改 setName 成员函数,在其中去掉超长的部分,确保数组不越界就可以了。

    可见,“隐藏”有利于程序的修改。

    “隐藏”机制还可以避免对对象的不正确操作。有的成员函数只是设计用来让同类的成员函数调用的,并不希望对外开放,因此就可以将它们声明为私有的,隐藏起来。

    现代软件开发绝大多数是合作完成的,一个程序员设计了一个类,可能被许多程序员使用。在设计类的时候,应当尽可能隐藏使用者不需要知道的实现细节,只留下必要的接口(即一些成员函数)来对对象进行操作,这样能够避免类的使用者随意使用成员函数和成员变量而导致错误。

    就像数字照相机的设计者会用外壳将内部的电路全部封装隐藏起来,用户不需要知道数字照相机的具体工作原理以及其中有哪些器件,只要能通过设计者留下的接口,即外壳上的各种按钮来使用照相机即可。如果把内部的电路、器件、开关都暴露给用户,那么外行用户很可能会把照相机弄坏。

更多...

加载中...