C++用纯虚函数实现协议委托的例子

??C++不像其他很多编程语言有接口、委托或者协议的概念,但是利用纯虚函数和C++多重继承的特性,我们也能实现接口、委托或协议要做的事情,下面的通过一个人设置闹钟然后被闹钟唤醒的例子来说明如何在C++中实现委托回调。

#include <iostream>
#include <vector>
#include <unistd.h>

using std::cout;
using std::endl;

// 前向声明
class Person;

// 用纯虚函数实现一个协议
// 实现该协议的类有一个被唤醒的行为
class Awakable {
public:
    // 纯虚函数
    virtual void beAwaken() = 0;
};

// 闹钟类
class AlarmClock {
public:
    // 委托方(谁委托了闹钟提供唤醒服务)
    Awakable *delegate;
    // 在指定时间后报警
    void alarmAfter(int) const;
};

void AlarmClock::alarmAfter(int seconds) const {
    sleep(seconds);
    cout << "叮咚 叮咚 叮咚" << endl;
    delegate->beAwaken();
}

// 人类(实现了Awakable协议可以被唤醒)
class Person : public Awakable {
public:
    // 启动闹钟并指定唤醒时间
    void rollClock(int);
    // 协议中的被唤醒的行为
    void beAwaken();
};

void Person::rollClock(int seconds) {
    cout << "人设置了闹钟" << seconds << "秒后响铃" << endl;
    // 在栈上创建闹钟对象
    AlarmClock ac;
    // 设置委托方
    ac.delegate = this;
    ac.alarmAfter(seconds);
}

void Person::beAwaken() {
    cout << "人被闹钟唤醒了" << endl;
}

int main(void) {
    // 创建人的对象
    Person *person = new Person;
    // 人启动闹钟设置5秒后被唤醒
    person->rollClock(5);
    return 0;
}

??程序写到这里,我相信已经不用更多的言语来解释如何实现协议委托了。其实各种编程语言的表象千变万化但是实质几乎没有区别。类似的功能在Java中用接口(interface)可以做到,在C#中可以使用委托(delegate),在Objective-C中可以使用协议(protocol),Swift中也有协议这个概念。但是如果你能够理解函数式编程的理念,还有更简单有效的方式就是使用Lambda函数,将一个回调函数直接作为参数传入一个函数或方法中,而Java(Java 8)和C#中都提供了Lambda表达式,OC中也有block来实现相同的功能。那么C++呢,别忘了C++中还有仿函数(函数对象)的概念,这些不都是一致的吗?当然,支持函数式编程范式的语言就更不用说了,就像JavaScript中可以把函数传入函数中,Swift不也是如此吗?

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-11 11:34:05

C++用纯虚函数实现协议委托的例子的相关文章

纯虚函数和虚函数

首先:强调一个概念定义一个函数为虚函数,不代表函数为不被实现的函数.定义他为虚函数是为了允许用基类的指针来调用子类的这个函数.定义一个函数为纯虚函数,才代表函数没有被实现.定义纯虚函数是为了实现一个接口,起到一个规范的作用,规范继承这个类的程序员必须实现这个函数.1.简介假设我们有下面的类层次: class A { public: virtual void foo() { cout<<"A::foo() is called"<<endl; } }; class

C++:纯虚函数与抽象类

5.4.3 纯虚函数和抽象类 纯虚函数是一个在基类中说明的虚函数,它在该基类中没有定义,但是要求在派生类中根据需要对它进行定义,或仍然说明为纯虚函数. 声明纯虚函数的一般格式是: virtual 函数类型 函数名(参数表)=0: 纯虚函数的作用是:在基类中为其派生类保留一个函数的名字,以便派生类根据需要对它进行重新定义.纯虚函数没有函数体,它最后面“=0 ”并不表示函数的返回值是0,它只是形式上的作用,告诉编译系统这是纯虚函数.纯虚函数不具有函数的功能,不能被调用. //应用举例 #includ

C++抽象类与纯虚函数实战举例

虚函数与纯虚函数请参考:http://blog.csdn.net/hackbuteer1/article/details/7558868 #include <iostream> using namespace std; class Animal { public:     virtual void eat()=0;     virtual void sleep()=0; }; class Cat:public Animal { public:     void eat()     {     

C++ 虚函数,纯虚函数的一些问题

#include <iostream> using namespace std; #define cendl cout << endl; class AA{//这是一个纯虚函数 public : virtual void fun() = 0; }; class A{ int data_a; public: A(){ data_a = 1; cout << "A " ; } A(int a){ data_a = a; cout << &qu

C++:抽象基类和纯虚函数的理解

转载地址:http://blog.csdn.net/acs713/article/details/7352440 抽象类是一种特殊的类,它是为了抽象和设计的目的为建立的,它处于继承层次结构的较上层. ⑴抽象类的定义: 称带有纯虚函数的类为抽象类. ⑵抽象类的作用: 抽象类的主要作用是将有关的操作作为结果接口组织在一个继承层次结构中,由它来为派生类提供一个公共的根,派生类将具体实现在其基类中作为接口的操作.所以派生类实际上刻画了一组子类的操作接口的通用语义,这些语义也传给子类,子类可以具体实现这些

c++中虚函数和纯虚函数定义

只有用virtual声明类的成员函数,使之成为虚函数,不能将类外的普通函数声明为虚函数.因为虚函数的作用是允许在派生类中对基类的虚函数重新定义.所以虚函数只能用于类的继承层次结构中. 一个成员函数被声明为虚函数后,在同一类族中的类就不能再定义一个非virtual的但与该虚函数具有相同的参数(包括个数和类型)和函数返回值类型的同名函数. 根据什么考虑是否把一个成员函数声明为虚函数? ①  看成员函数所在的类是否会作为基类 ② 看成员函数在类的继承后有无可能被更改功能,如果希望更改其功能的,一般应该

虚函数和纯虚函数的作用与区别

http://blog.csdn.net/xwpc702/article/details/8670025 虚函数为了重载和多态的需要,在基类中是有定义的,即便定义是空,所以子类中可以重写也可以不写基类中的此函数!纯虚函数在基类中是没有定义的,必须在子类中加以实现,很像java中的接口函数!虚函数引入原因:为了方便使用多态特性,我们常常需要在基类中定义虚函数.class Cman{public:virtual void Eat(){……};void Move();private:};class C

C++ 虚函数与纯虚函数 浅析

[摘要] 本文首先简述虚函数与纯虚函数的定义,然后分析比较两者的区别与联系(DWS). [正文] 1)虚函数与纯虚函数有什么区别? 虚函数,不代表函数为不被实现的函数,为了允许用基类的指针来调用子类的这个函数:允许被其子类重新定义的成员函数. 纯虚函数,才代表函数没有被实现,为了实现一个接口,起到一个规范的作用,规范继承这个类的程序员必须实现这个函数. 2)虚就虚在所谓"推迟联编"或者"动态联编"上,一个类函数的调用并不是在编译时刻被确定的,而是在运行时刻被确定的.

C++虚函数和纯虚函数

只有用virtual声明类的成员函数,使之成为虚函数,不能将类外的普通函数声明为虚函数.因为虚函数的作用是允许在派生类中对基类的虚函数重新定义.所以虚函数只能用于类的继承层次结构中. 一个成员函数被声明为虚函数后,在同一类族中的类就不能再定义一个非virtual的但与该虚函数具有相同的参数(包括个数和类型)和函数返回值类型的同名函数. 根据什么考虑是否把一个成员函数声明为虚函数? ①  看成员函数所在的类是否会作为基类 ② 看成员函数在类的继承后有无可能被更改功能,如果希望更改其功能的,一般应该