设计模式的学习

OO:Object-Oriented

面向对象

---

《Head First设计模式》

这本书是用java写的,我是写C++的,用C++来写相关的代码

---

p2(第二页)

#ifndef DUCK_H
#define DUCK_H

/**
 * @brief The Duck class    鸭子类
 */
class Duck
{
public:
    Duck();
    /**
     * @brief quack 呱呱叫
     * 所有的鸭子都会呱呱叫
     * 所以在父类中实现这部分代码
     */
    void quack();
    /**
     * @brief swim  游泳
     * 所有的鸭子都会游泳
     * 所以在父类中实现这部分代码
     */
    void swim();
    /**
     * @brief display   展示鸭子的外观
     * 每种鸭子的外观特点不同,需要在子类中进行实现
     * 所以这里是虚函数
     */
    virtual void display();
};

#endif // DUCK_H
#ifndef MALLARDDUCK_H
#define MALLARDDUCK_H

#include "duck.h"
/**
 * @brief The MallardDuck class 绿头鸭
 */
class MallardDuck : public Duck
{
public:
    MallardDuck();
    virtual void display();
};

#endif // MALLARDDUCK_H
#ifndef MALLARDDUCK_H
#define MALLARDDUCK_H

#include "duck.h"
/**
 * @brief The MallardDuck class 绿头鸭
 */
class MallardDuck : public Duck
{
public:
    MallardDuck();
    virtual void display();
};

#endif // MALLARDDUCK_H

---

现在需要添加需求:一部分鸭子会飞。

如果直接在基类中添加fly(),那么意味着所有的鸭子都会飞。(所有的子类都拥有了fly() )

而橡皮鸭子是不会飞的。

---

当涉及“维护”时,为了复用目的而使用继承,并不是完美的。

---

如果子类把fly()方法覆盖掉呢?

可以,但如果再加入一种不会飞的鸭子,那么需要重新再覆盖fly()方法。

void fly()
{
    //覆盖,什么事都不做
}

每多一种鸭子,就需要覆盖一次fly()方法。

------

而有些鸭子不会飞,也不会叫。

---

希望找到一种建立软件的方法,让我们可以用一种对既有的代码影响最小的方式来修改软件。

这样我们就可以花较少时间重做代码,而让程序去做更酷的事。

---

继承不能很好地解决问题,因为鸭子的行为在子类中不断地改变,并且让所有的子类都有这些行为是不恰当的。

Flyable和Quackable接口一开始似乎还不错,但继承接口无法达到代码的复用。

当需要修改某个行为时,需要在定义此行为的类中修改它。

---

设计原则: 

找出应用中可能需要变化之处,把它们独立出来,不要和不需要变化的代码混在一起。

---

努力让系统中的某部分改变不会影响其他部分。

---

设计原则:

针对接口编程,而不是针对实现编程。

---

简单的多态例子:

有一个抽象类Animal,两个具体的实现:Dog和Cat继承自Animal

针对实现编程:

Dog d = new Dog();
d.bark();

针对接口编程:

Animal * animal = new Dog();
animal.makeSound();

子类实例化的动作不再需要在代码中硬编码,而是在运行时才指定具体实现的对象,

进一步封装:

a = getAnimal();
a.makeSound();

我们不关心实际的子类型是什么,我们只关心它知道如何正确地调用makeSound()。

---

#ifndef FLYBEHAVIOR_H
#define FLYBEHAVIOR_H
#include <QDebug>
/**
 * @brief The FlyBehavior class 飞行行为
 */
class FlyBehavior
{
public:
    FlyBehavior()
    {

    }
    virtual void fly()
    {

    }
};

/**
 * @brief The FlyWithWings class    用翅膀飞行
 */
class FlyWithWings : public FlyBehavior
{
public:
    FlyWithWings()
    {

    }

    virtual void fly()
    {
        qDebug()<<"用翅膀飞行";
    }
};

/**
 * @brief The FlyNoWay class    不会飞行
 */
class FlyNoWay : public FlyBehavior
{
public:
    FlyNoWay()
    {

    }

    virtual void fly()
    {
        qDebug()<<"不会飞行";
    }
};

#endif // FLYBEHAVIOR_H
#ifndef FLYBEHAVIOR_H
#define FLYBEHAVIOR_H
#include <QDebug>
/**
 * @brief The FlyBehavior class 飞行行为
 */
class FlyBehavior
{
public:
    FlyBehavior()
    {

    }
    virtual void fly()
    {

    }
};

/**
 * @brief The FlyWithWings class    用翅膀飞行
 */
class FlyWithWings : public FlyBehavior
{
public:
    FlyWithWings()
    {

    }

    virtual void fly()
    {
        qDebug()<<"用翅膀飞行";
    }
};

/**
 * @brief The FlyNoWay class    不会飞行
 */
class FlyNoWay : public FlyBehavior
{
public:
    FlyNoWay()
    {

    }

    virtual void fly()
    {
        qDebug()<<"不会飞行";
    }
};

#endif // FLYBEHAVIOR_H

关键在于:

鸭子现在会将飞行和呱呱叫的动作“委托”别人处理,而不是使用定义在Duck类或子类内的呱呱叫和飞行方法。

#ifndef DUCK_H
#define DUCK_H

#include "flybehavior.h"
#include "quackbehavior.h"
/**
 * @brief The Duck class    鸭子类
 */
class Duck
{
public:
    Duck();
    FlyBehavior * flyBehavior;
    QuackBehavior * quackBehavior;
    void performQuack()
    {
        quackBehavior->quack();
    }
    void performFly()
    {
        flyBehavior->fly();
    }
    /**
     * @brief display   展示鸭子的外观
     * 每种鸭子的外观特点不同,需要在子类中进行实现
     * 所以这里是虚函数
     */
    virtual void display()
    {

    }
};

#endif // DUCK_H
#ifndef MALLARDDUCK_H
#define MALLARDDUCK_H

#include "duck.h"
/**
 * @brief The MallardDuck class 绿头鸭
 */
class MallardDuck : public Duck
{
public:
    MallardDuck();
    virtual void display()
    {

    }
};

#endif // MALLARDDUCK_H
#include "mallardduck.h"

MallardDuck::MallardDuck()
{
    flyBehavior = new FlyWithWings();
    quackBehavior = new MuteQuack();
}
    MallardDuck mallardDuck;
    mallardDuck.performFly();
    mallardDuck.performQuack();

---

通过基类的指针析构子类时,会出现内存泄漏吗?

如果不希望内存泄漏,析构函数必须定义为虚函数。

---

设计原则:

多用组合,少用继承。

使用组合建立系统具有很大的弹性,不仅可将算法族封装成类,更可以“在运行时动态地改变行为”。

策略模式:

定义了算法族,分别封装起来,让它们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。

上面的鸭子示例中:可以在运行时动态地切换鸭子相关的行为(算法)(用翅膀飞,不会飞)

------

设计是一门艺术,总是有许多可取舍的地方。

设计大师关心的是建立弹性的设计,可以维护,可以应付变化。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lpl还在学习的路上

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值