C++ 类之间的互相调用
这几天做C++11的线程池时遇到了一个问题,就是类A想要调用类B的方法,而类B也想调用类A的方法
这里为了简化起见,我用更容易理解的观察者模式向大家展开陈述
观察者模式:在对象之间定义一对多的依赖,这样一来,当一个对象改变状态时,依赖它的对象都会收到通知,并自动更新

观察者模式中有一个subject和observer
observer向subject注册成为一个观察者
当subject发生改变时,它通知所有的观察者
当一个observer不想作为观察者时,它会向subject发出请求,将自己从观察者中除名
注意,在这里是存在一个互相调用的
subject肯定需要知道observer的方法,这样它才能在状态发生改变时调用observer的方法通知他们
而当一个observer想要将自己从观察者中除名的时候,它需要保留一个subjet的引用,并让subject调用remove方法将自己除名
为了简化起见
在这里的类图如下

在java,我们可以这样实现
import java.util.ArrayList;
class Subject { public void change() { for (Observer x :observerList) {
x.Show();
}
} public void register(Observer o) {
observerList.add(o);
} public void Remove(Observer o) {
observerList.remove(o);
} private ArrayList<Observer> observerList=new ArrayList<Observer>();
} class Observer {
public void Show() {
System.out.println("I konw The Subject is changed");
} public Observer(Subject s) {
subject = s;
} public void Remove() {
subject.Remove(this);
} private Subject subject;
} public class Observertry { public static void main(String[] args) {
Subject s = new Subject();
Observer o = new Observer(s);
s.register(o);
s.change(); }
}
运行结果

而在C++中
如果我们在main.cpp中编写出以下代码
#include <iostream>
#include <string>
#include <vector>
using namespace std;
class Observer;
class Subject;
class Observer
{
public: Observer(Subject *s)
{
subject = s;
}
void Remove(Subject *s)
{
s->Remove(this);
}
void Show()
{
cout << "I konw Subject is change" << endl;
}
private:
Subject *subject;
};
class Subject
{
public:
void change()
{
for (vector<Observer*>::iterator it = observerlist.begin(); it != observerlist.end(); it++)
{
(*it)->Show();
} }
void Remove(Observer *o)
{
observerlist.erase(find(observerlist.begin(), observerlist.end(), o));
}
void Register(Observer *o)
{
observerlist.push_back(o);
}
private:
vector<Observer*> observerlist; };
int main()
{
Subject s;
Observer o(&s);
s.Register(&o);
s.change();
system("pause");
}
会发现这段代码无法编译通过
在vs2013中会有以下error

这是因为虽然有类的成员的前向声明
但你仅可以定义指向这种裂隙的指针或引用,可以声明但不能定义以这种不完全类型或者返回类型的参数
而这里你想要在Observer类里调用subject的方法,而subject是在Observer的后面声明定义的,所以无法调用subject的方法
而C++是没有对类的函数的前向声明的
所以我们要有一个方法,让我们在声明类Subject时能看到类Observer的声明
而在声明类Observer时,能看到类Subject的声明
所以我们想到将Subject和Observer分别放到两个文件中去
所以我们有了如下尝试
subject.h
#pragma once
#include "Observer.h"
#include <iostream>
#include <vector> class Subject
{
public:
void change();
void Remove(Observer *o);
void Register(Observer *o);
std::vector<Observer*> observerlist;
};
observer.h
#pragma once
#include <iostream>
#include "Subject.h"
using namespace std;
class Subject; class Observer
{
public:
Observer(Subject *s); void Remove(Subject *s);
void Show();
Subject *subject;
};
但这一次依旧无法通过编译
因为我们这里出现了头文件的互相包含
subject.h中包含了observer.h
observer.h中包含了subject.h
所以正确的方法是把其中的一个的include放到相应的实现文件中即cpp文件中
代码如下
subject.h
#pragma once
#include "Observer.h"
#include <iostream>
#include <vector> class Subject
{
public:
void change();
void Remove(Observer *o);
void Register(Observer *o);
std::vector<Observer*> observerlist;
};
subject.cpp
#include "Subject.h" void Subject::change()
{
for (vector<Observer*>::iterator it = observerlist.begin(); it != observerlist.end(); it++)
{
(*it)->Show();
}
} void Subject::Remove(Observer *o)
{
observerlist.erase(find(observerlist.begin(), observerlist.end(), o));
}
void Subject::Register(Observer *o)
{
observerlist.push_back(o);
}
observer.h
#pragma once
#include <iostream> using namespace std;
class Subject; class Observer
{
public:
Observer(Subject *s); void Remove(Subject *s);
void Show();
Subject *subject;
};
observer.cpp
#include "Observer.h"
#include "Subject.h" Observer::Observer(Subject *s)
{
subject = s;
} void Observer::Remove(Subject *s)
{
s->Remove(this);
}
void Observer::Show()
{
cout << "I know Subject is changed" << endl;
}
我们将#include “Subject.h”放到了observer.cpp中
这样就在observer的实现中就可以看到Subject的声明,进而调用subject的Remove方法,有不会引起互相包含的问题了
运行结果如下

C++ 类之间的互相调用的更多相关文章
- Java 类与类之间的调用
方法1. 新建一个类. 然后在调用类中先进行被调用类实例化,然后通过实例化的对象访问. 例如: //先定义一个类 import static java.lang.System.out; public ...
- (转) C++中基类和派生类之间的同名函数的重载问题
下面有关派生类与基类中存在同名函数 fn: class A { public: void fn() {} void fn(int a) {} }; class B : public A { publi ...
- JAVA和C/C++之间的相互调用。
在一些Android应用的开发中,需要通过JNI和 Android NDK工具实现JAVA和C/C++之间的相互调用. Java Native Interface (JNI)标准是java平台的一部分 ...
- 图解UML类与类之间的六中关系
大话设计模式上的一个图,我用EA画出来的: UML中的6大关系相关英文及音标: 依赖关系 dependency [di'pendənsi] 关联关系 association [ə,səuʃi' ...
- Java类之间的关联关系(转载)
Java类之间的关联关系 UML类图中的关系分为四种:泛化.依赖.关联.实现:关联关系又可以细化为聚合和组合. 一.泛化(Generalization) 泛化是父类和子类之间的关系,子类继承父类的所有 ...
- oc之封装与类之间的关系
1. 面向对象的特征-封装? 封装: 现实生活中的封装: 将很多的小东西 塞在1个大口袋里面. 好处: a. 对外部屏蔽. b. 方便管理. 代码的封装: 函数/方法 就是1种封装的体现: 将一段代码 ...
- java 类与类之间的关系 及uml图
类与接口之间的关系 : 继承 类与类之间的关系 :继承关系 包含关系 类与对象之间的关系 : 实例 UML 类图中类与类之间的关系: 泛化关系(generalization) 关联关系(associ ...
- 全面解释java中StringBuilder、StringBuffer、String类之间的关系
StringBuilder.StringBuffer.String类之间的关系 java中String.StringBuffer.StringBuilder是编程中经常使用的字符串类,在上一篇博文中我 ...
- 【47】java的类之间的关系:泛化、依赖、关联、实现、聚合、组合
java的类之间的关系:泛化.依赖.关联.实现.聚合.组合 泛化: • 泛化关系(Generalization)也就是继承关系,也称为"is-a-kind-of"关系,泛化关系用于 ...
随机推荐
- 评价指标1--F1值和MSE
1,F1=2*(准确率*召回率)/(准确率+召回率) F1的值是精准率与召回率的调和平均数.F1的取值范围从0到1的数量越大,表明实现越理想. Precision(精准率)=TP/(TP+FP) Re ...
- ES6的Promise对象
http://es6.ruanyifeng.com/#docs/promise Promise 对象 Promise 的含义 基本用法 Promise.prototype.then() Promise ...
- Hadoop namenode启动瓶颈分析
NameNode启动过程详细剖析 NameNode中几个关键的数据结构 FSImage Namenode会将HDFS的文件和目录元数据存储在一个叫fsimage的二进制文件中,每次保存fsimage之 ...
- #6435. 「PKUSC2018」星际穿越
考场上写出了70分,现在填个坑 比较好写的70分是这样的:(我考场上写的贼复杂) 设\(L(i)=\min_{j=i}^nl(j)\) 那么从i开始向左走第一步能到达的就是\([l(i),i-1]\) ...
- 使用Keil下载单独的Hex文件到单片机内
前言 初学STM32时,是通过串口1把Hex文件下载进STM32的,需要一个串口模块,而且还要设置BOOT0和BOOT1电平,然后通过FlyMcu软件进行下载,这也是一种不错的方法,这里我要介绍的是使 ...
- 微信小程序如何检测接收iBeacon信号
前话 微信小程序开发带着许多坑,最近就遇到了个需求,检测iBeacon来进行地点签到. (╯▔皿▔)╯ 微信小程序对于iBeacon的文档也写的十分精简,只简单介绍了每个接口的作用,这就导致我以为简单 ...
- TortoiseGit版本库中某个文件显示问号或叹号的问题解决办法
这是一个怪问题,原因就是文件名大小写与版本库管理的大小写不一致. 解决办法: 1.先把文件夹中的物理文件名改为版本库浏览器中显示的文件名(版本库浏览器中的文件名不知道怎么改),改了以后这个文件图标就变 ...
- JavaWeb项目学习教程(2) 系统数据库设计
最开始本来想写一个管理系统,因为考虑到期末来临,我女朋友就可以看着教程然后学一些东西,然后可以自己慢慢手敲代码.但无奈自己也太懒,两个月过后,我才开始继续写这个博客,而现在我都已经开学了.不过博客还是 ...
- 自己编写的:centos6.6上编译安装apache2.4+php5.6+mysql5.6【亲自】
在centos6.6上安装apache2.4+php5.6+mysql5.6 关于wget的安装 将之前装系统的.iso文件挂载到光驱 由于我在/home/jinnan/下建立了一个cdrom文件夹 ...
- IDEA配置maven中央库
分两步: STEP :配置maven: STEP :配置IDEA.区分默认配置和项目级配置. STEP 1:maven中央库配置 国内常用的maven库主要是阿里云maven库.华为云maven. 其 ...