C++代码训练营 | 多样的星空

前两篇中,我们分别介绍了面向对象的封装和继承两个特性,今天我们来说最后一个特性:多态。

什么是多态

用一句话来概括多态:允许父类的指针指向子类对象。

为什么要用父类的指针去指向子类对象呢?我们想象一个最简单的场景,如果我们需要一个函数的参数是可变数据类型,那如何实现呢?C++是不允许模糊数据类型存在的,这个需求听起来几乎不可能实现。不过有了多态,我们可以把参数类型设置为父类的指针类型,这样在参数传递的时候我们就可以传递这个父类的任意一个子类的对象了。听起来比较乱,我们用一个例子来解释。

先看看下面这段代码:

#include <iostream>

using namespace std;

class Animal
{
public:
    virtual void Run() = 0;
    virtual void Cry() = 0;
};

class Dog : public Animal
{
    void Run()
    {
        cout << "Dog is running ..." << endl;
    }

    void Cry()
    {
        cout << "Dog is crying ..." << endl;
    }
};

class Cat : public Animal
{
    void Run()
    {
        cout << "Cat is running ..." << endl;
    }

    void Cry()
    {
        cout << "Cat is crying ..." << endl;
    }
};

void Functions(Animal* pAnimal)
{
    pAnimal->Run();
    pAnimal->Cry();
}

int main()
{
    Dog dog;
    Cat cat;

    Functions(&dog);
    Functions(&cat);
}

运行结果:

先看Functions()函数,它的参数是一个Animal类指针,Animal是一个抽象类,派生了两个子类:Dog和Cat。在main函数中,我们可以很方便地把这两个子类的对象传给Animal指针。这就是多态。

多态的背后隐藏着伟大的设计模式思想,希望大家慢慢体会。

抽象类

Animal是个抽象类。它的特点是:

  • 包含虚函数
  • 不能实例化

为什么要用抽象类呢?它的意义仅仅是定义成员函数的外观。在这个例子中,我们定义了两个虚函数:Run()和Cry()。有了这样一个抽象类,我们才能保证Dog和Cat两个子类都拥有一模一样的两个成员函数。否则,Functions()函数就没法正常工作了。

星空问题的新需求

回到星空那段代码,假如我们这个项目需要A,B,C,三个人同时开发。A开发点状星星,B开发矩形的星星,C开发X形的星星。我们要确保三个人能同时开发,完成后能很容易的把三份代码加入项目中,该怎么办呢?

先分析一下代码:

class Star
{
public:
    Star(){}
    ~Star(){}

    void Init();
    void Move();

protected:
    void Draw();
    void NewPos();
    void Remove();

    double  m_x = 0;
    int     m_y;
    double  m_step;
    int     m_color;
};

Star类中,影响星星形状的成员函数是Draw()和Remove(),我们可以利用多态把这两个函数抽象出来成为一个抽象类,之后让A,B,C三个人分别实现一个子类,这样就OK了。

来看看代码吧:

#include <graphics.h>
#include <time.h>
#include <conio.h>

#define SCREEN_WIDTH    1024
#define SCREEN_HEIGHT   768
#define MAXSTAR         400

class StarType
{
public:
    virtual void Draw(int x, int y, int color) = 0;
    virtual void Remove(int x, int y) = 0;
};

class PointStar : public StarType
{
    void Draw(int x, int y, int color)
    {
        putpixel((int)x, y, color);
        setcolor(color);
        circle(x, y, 1);
    }

    void Remove(int x, int y)
    {
        putpixel((int)x, y, 0);
        setcolor(0);
        circle(x, y, 1);
    }
};

class RectStar : public StarType
{
    void Draw(int x, int y, int color)
    {
        putpixel((int)x, y, color);
        setcolor(color);
        rectangle(x - 1, y - 1, x + 1, y + 1);
    }

    void Remove(int x, int y)
    {
        putpixel((int)x, y, 0);
        setcolor(0);
        rectangle(x - 1, y - 1, x + 1, y + 1);
    }
};

class XStar : public StarType
{
    void Draw(int x, int y, int color)
    {
        settextcolor(color);
        outtextxy(x, y, _T("x"));
    }

    void Remove(int x, int y)
    {
        settextcolor(0);
        outtextxy(x, y, _T("x"));
    }
};

class Star
{
public:
    Star(){}
    ~Star(){}

    void Init();
    void Init(StarType* pStarType);
    void Move();

protected:
    void NewPos();

    double  m_x = 0;
    int     m_y;
    double  m_step;
    int     m_color;

    StarType* m_pStarType;
};

void Star::Init()
{
    if (m_x == 0)
    {
        m_x = rand() % SCREEN_WIDTH;
    }
    else
    {
        m_x = 0;
    }

    m_y = rand() % SCREEN_HEIGHT;
    m_step = (rand() % 5000) / 1000.0 + 1;
    m_color = (int)(m_step * 255 / 6.0 + 0.5);  // 速度越快,颜色越亮
    m_color = RGB(m_color, m_color, m_color);
}

void Star::Init(StarType* pStarType)
{
    this->Init();
    m_pStarType = pStarType;
}

void Star::Move()
{
    m_pStarType->Remove(m_x, m_y);

    NewPos();

    m_pStarType->Draw(m_x, m_y, m_color);
}

void Star::NewPos()
{
    m_x += m_step;
    if (m_x > SCREEN_WIDTH)
        this->Init();
}

void main()
{
    srand((unsigned)time(NULL));
    
    initgraph(SCREEN_WIDTH, SCREEN_HEIGHT); 

    Star star[MAXSTAR];

    PointStar pointStar;
    RectStar rectStar;
    XStar xStar;
    for (int i = 0; i < MAXSTAR; i++)
    {
        switch (i % 3)
        {
        case 0:
            star[i].Init(&pointStar);
            break;
        case 1:
            star[i].Init(&rectStar);
            break;
        case 2:
            star[i].Init(&xStar);
            break;
        default:
            break;
        }
    }

    while (!kbhit())
    {
        for (int i = 0; i < MAXSTAR; i++)
        {
            star[i].Move();
        }
    
        Sleep(50);
    }

    closegraph();
}

效果如下:

程序中,我们定义了一个抽象类StarType,之后三个类分别派生与这个抽象类。

我们重载了Star类的Init()方法,当我们要实现不同样子的星星时,就把相应的对象地址通过Init()方法传给Star类保存在m_pStarType属性中。当执行Move函数时,就会通过这个属性调用不同的方法来完成。是不是很神奇呢?

这其实是一种设计模式,假如我们有一天需要再多加几种星星,只需要再多实现几个StarType的派生类就行了,其他代码几乎不用改变。

这部分重在理解编程的思想,希望大家多多思考,有问题欢迎讨论。

到此,我们这个项目就告一段落了,对于一个稍大的项目而言,文件划分非常重要。我会将划分好的代码放在GitHub上,欢迎大家下载源码学习。

我是天花板,让我们一起在软件开发中自我迭代。
如有任何问题,欢迎与我联系。


上一篇:C++代码训练营 | 另一片星空

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,439评论 25 707
  • 1.import static是Java 5增加的功能,就是将Import类中的静态方法,可以作为本类的静态方法来...
    XLsn0w阅读 1,209评论 0 2
  • 1.面向对象的程序设计思想是什么? 答:把数据结构和对数据结构进行操作的方法封装形成一个个的对象。 2.什么是类?...
    少帅yangjie阅读 4,987评论 0 14
  • 面向对象主要针对面向过程。 面向过程的基本单元是函数。 什么是对象:EVERYTHING IS OBJECT(万物...
    sinpi阅读 1,041评论 0 4
  • 早上, 58.2kg 上午 1.泡脚 2.按摩 3.15斤坐提,三个方向各30个 4.顶墙一组 5.直膝推墙一组 ...
    慕洋阅读 217评论 0 0