• C++继承和组合(无师自通)

    在面向对象语言中,类继承可用于描述派生类的类型是基类的一种特殊情况的事实,并建立相应的模型。实际上,类应该被视为可从中创建的所有对象的集合。因为派生类是基类的特例,所以对应于派生类的对象集合将是对应于基类的对象集合的子集。因此,派生类的每个对象也是基类的一个对象。换句话说,每个派生类对象都是一个(Is-a)基类对象。

    每当某个类包含另一个类的对象作为其成员变量之一时,就会出现类组合。组合在两个类之间建立了(Has-a)的关系。

    由于派生类继承了其基类的所有成员,派生类实际上包含其基类的一个对象。正因为如此,可能在某些需要组合的地方也可以使用继承。

    现在来看一个示例。假设有一个程序需要能代表某个人的数据,比如说这个人的姓名和街道地址。街道地址可能由两行组成:

    123 Main Street
    Hometown, 12345

    现在假设有一个代表街道地址的类:

    class StreetAddress
    {
        private:
            string line1, line2;
        public:
            void setLine1(string);
            void setLine2(string);
            string getLine1();
            string getLine2();
    };

    因为一个人的数据包含一个姓名和一个街道地址,所以正确表示一个人的数据的类将使用以下方式的组合:

    class PersonData
    {
        private:
            string name;
            StreetAddress address;
        public:
            ...
    };

    PemmData 的类声明中忽略了其余部分,因为那些与我们要阐述的主旨无关。

    在这里可以使用继承而不是组合来定义这个类。例如,可以定义一个类 PersonData1,如下所示:

    class PersonData1:public StreetAddress
    {
        private:
            string name;
        public:
    };

    虽然这个新的定义能够正确编译,但从概念上讲这其实是错误的,因为它将一个人的数据视为一种特殊的街道地址,而事实并非如此。这种类型的设计概念错误可能会导致程序理解困扰并难以维护。

    所以,更好的设计做法是,尽可能优先选择组合而不是继承。这样做还有一个原因是,继承打破了基类的封装,因为它将基类的受保护成员暴露给了派生类的方法。

    现在来看一个使用继承比组合更有意义的示例。假设有一个 dog 类,代表了所有狗的集合。每个 Dog 对象都有一个类型为 double 的成员变量 weight 和一个成员函数 voidbark(),该类的示例如下:

    class Dog
    {
        protected:
            double weight;
        public:
            Dog(double w)
            { weight = w; }
            virtual void bark() const
            {
                cout << "I am dog weighing " << weight << " pounds." << endl;
            }
    };

    这个类还有一个构造函数,允许 Dog 对象被初始化。请注意,以上示例中已经声明了一个虚成员函数 bark(),以允许它在派生类中被覆盖。

    假设需要一个代表所有牧羊犬集合的类。由于每只牧羊犬都是狗,因此从 Dog 类中派生出新的 SheepDog 类是有意义的。这样,SheepDog 对象将继承 Dog 类的每个成员。除了具有狗所具有的各种特征之外,每只牧羊犬还应该具有区别于其他犬种的特殊特征,例如,有一个整数成员 numberSheep 指示牧羊犬被训练看护的绵羊的最大数量。另外,牧羊犬的吠叫方式可能不同于普通犬种,也许要适应看护羊群的需要。这可以通过覆盖 Dog 类的 bark() 成员函数来解决。

    class SheepDog:public Dog
    {
        private:
            int numberSheep;
        public:
            SheepDog(double w, int nSheep) : Dog(w)
            {
                numberSheep = nSheep;
            }
            virtual void bark() const override
            {
                cout << "I am a sheepdog weighing " << weight << " pounds \nand guarding " << numberSheep << " sheep." << endl;
            }
    };

    为了演示该类,可以建立一个狗的矢量,矢量中的一些狗就是牧羊犬。为了规避矢量不能拥有两种不同类型的事实,可以使用指向 Dog 的指针的矢量。前面讲过,一个指向基类(在本示例中即 Dog 类)的指针也可以指向任何派生类对象(在本示例中为 SheepDog)。因此,可以创建一个指向 Dog 的指针矢量,并且其中一些指针指向 Dog 对象,而另一些指针则指向 SheepDog 对象。

    vector<shared_ptr<Dog>> kenne1
    {
        make_shared<Dog>(40.5),
        make_shared<SheepDog>(45.3, 50),
        make_shared<Dog>(24.7)
    };

    最后,可以使用一个循环来调用矢量中每个 Dog 对象的 bark() 成员函数:

    for (int k = 0; k < 3; k++)
    {
        cout << k+1 << ": ";
        kennel[k]->bark();
    }

    由于多态性,并且因为 bark() 函数被声明为虚函数,所以循环内的同一行代码对普通狗将调用原始的 bark() 函数,而对牧羊犬则会调用派生类 SheepDog 中的特殊 bark() 函数。完整的程序为:

    // This program demonstrates the Is-A relation in inheritance.
    #include <iostream>
    #include <memory>
    #include <vector>
    
    using namespace std;
    // Base class
    class Dog
    {
        protected:
            double weight;
        public:
            Dog(double w)
            {
                weight = w;
            }
            virtual void bark() const
            {
                cout << "I am a dog weighing " << weight << " pounds." << endl;
            }
    };
    
    // A SheepDog is a special type of Dog
    class SheepDog :public Dog
    {
            int numberSheep;
        public:
            SheepDog(double w, int nSheep) : Dog(w)
            {
                numberSheep = nSheep;
            }
            void bark() const override
            {
                cout << "I am a sheepdog weighing " << weight << " pounds and guarding " << numberSheep << " sheep." << endl;
            }
    };
    
    int main()
    {
        // Create a vector of dogs
        vector<shared_ptr<Dog>> kennel
        {
            make_shared<Dog>(40.5),
            make_shared<SheepDog>(45.3, 50),
            make_shared<Dog>(24.7)
        };
        // Walk by each kennel and make the dog bark
        for (int k = 0; k < kennel.size(); k++)
        {
            cout << k + 1 << ": ";
            kennel[k]->bark();
        }
        return 0;
    }

    程序输出结果:

    1: I am a dog weighing 40.5 pounds.
    2: I am a sheepdog weighing 45.3 pounds and guarding 50 sheep.
    3: I am a dog weighing 24.7 pounds.

    在本示例中,继承是一个比组合更好的选择,因为使用组合就等于说一只牧羊犬有一只(Has-a)狗,而不是说一只牧羊犬是一只(Is-a)狗。

    有些作者认为在类之间存在第三种关系,即:使用实现关系。基本上,一个类如果调用第二个类的对象的成员函数,则称它使用了第二个类的实现。

    如何才能知道何时该使用继承,何时该使用组合呢?

    假设有一个现有的类 C1,并且需要为另一个类 C2 编写一个定义,而 C2 需要一个关联的 C1 对象的服务。那么,究竟是需要从 C1 派生 C2,还是应该给 C2 —个 C1 类型的成员变量?一般来说,应该优选组合而不是继承。

    为了帮助确定继承是否合适,可以提出以下问题:

    • 将 C2 对象设想成 C1 对象的特殊类型是否自然?如果是,那么应该使用继承。
    • C2 类对象是否需要在 C1 类对象使用的地方使用?例如,C2 对象是否需要被传递给函数,而该函数釆用的引用形参为 C1 类型或指向 C1 的指针?如果是,那么应该使 C2 成为 C1 的派生类。

更多...

加载中...