C++类继承继承5——构造函数与拷贝控制

news2024/11/27 14:44:39

构造函数与拷贝控制

和其他类一样,位于继承体系中的类也需要控制当其对象执行一系列操作时发生什么样的行为,这些操作包括创建、拷贝、移动、赋值和销毁。

如果一个类(基类或派生类)没有定义拷贝控制操作,则编译器将为它合成一个版本。当然,这个合成的版本也可以定义成被删除的函数。

虚析构函数

继承关系对基类拷贝控制最直接的影响是基类通常应该定义一个虚析构函数,这样我们就能动态分配继承体系中的对象了。

如前所述,当我们delete一个动态分配的对象的指针时将执行析构函数。如果该指针指向继承体系中的某个类型,则有可能出现指针的静态类刑与被删除对象的动态类型不符的情况。

class Quote
{
};
class Bulk_quote:public Quote
{
};

例如,如果我们delete一个Quote*类型的指针,则该指针有可能实际指向了一个Bulk_quote类型的对象如果这样的话,编译器就必须清楚它应该执行的是Bulk guote的析构函数。和其他函数一样,我们通过在基类中将析构函数定义成虚函数以确保执行正确的析构函数版本,

class Quote {
public:
// 如果我们删除的是一个指向派生类对象的基类指针,则需要虚析构函数
virtual ~Quote() = default; //动态绑定析构函数
};

和其他虚函数一样,析构函数的虚属性也会被继承。因此,无论Quote的派生类使用合成的析构函数还是定义自己的析构函数,都将是虚析构函数。

只要基类的析构函数是虚函数,就能确保当我们delete基类指针时将运行正确的析构函数版本:

#include<iostream>
using namespace std;
class Quote
{
public :
	virtual ~Quote()
	{
		cout << "基类析构" << endl;
	}
};
class Bulk_quote :public Quote
{
	~Bulk_quote()
	{
		cout << "派生类析构" << endl;
	}
};
int main()
{
	Quote* itemP = new Quote; // 静态类型与动态类型一致
	delete itemP; //调用Quote的析构函数
	itemP = new Bulk_quote; // 静态类型与动态类型不一致
	delete itemP; // 调用Bulk_quote的析构函数
}


如果基类的析构函数不是虚函数,则delete一个指向派生类对象的基类指针将产生未定义的行为。

#include<iostream>
using namespace std;
class Quote
{
public :
  ~Quote()
	{
		cout << "基类析构" << endl;
	}
};
class Bulk_quote :public Quote
{
	~Bulk_quote()
	{
		cout << "派生类析构" << endl;
	}
};
int main()
{
	Quote* itemP = new Quote; // 静态类型与动态类型一致
	delete itemP; //调用Quote的析构函数
	itemP = new Bulk_quote; // 静态类型与动态类型不一致
	delete itemP; // 调用Bulk_quote的析构函数
}

如果一个类需要析构函数,那么它也同样需要拷贝和赋值操作。

基类的析构函数并不遵循上述准则,它是个重要的例外。

一个基类总是需要析构函数,而且它能将析构函数设定为虚函数。

此时该析构函数为了成为虚函数而令内容为空,我们显然无法由此推断该基类还需要赋值运界符或拷贝构造函数。

虚析构函数将阻止合成移动操作

基类需要一个虚析构函数这一事实还会对基类和派生类的定义产生另外一个间接的影响:

如果一个类定义了析构函数,即使它通过=default的形式使用了合成的版本,编译器也不会为这个类合成移动操作。

合成拷贝控制与继承

基类或派生类的合成拷贝控制成员的行为与其他合成的构造函数、赋值运算符或析构函数类似:它们对类本身的成员依次进行初始化、赋值或销毁的操作。此外,这些合成的成员还负责使用直接基类中对应的操作对一个对象的直接基类部分进行初始化、赋值或销毁的操作。

例如,

class Quote
{
public:
string bookNo;
int price=0;
virtual ~Quote(){}
};
class Disc_quote:public Quote
{
public:
int quantity=0;
int discount=0;
};
class Bulk_quote:public Disc_quote
{
};
  1. 合成的Bulk_quote默认构造函数运行Disc_quote的默认构造函数,后者又运行Quote的默认构造函数。
  2. Quote 的默认构造函数将bookNo成员默认初始化为空字符串,同时使用类内初始值将price初始化为0。
  3. Quote的构造函数完成后,继续执行Disc_quote的构造函数,它使用类内初始值初始化qty和discount。
  4. Disc_quote的构造函数完成后,继续执行Bulk quote的构造函数,但是它什么具体工作也不做。

类似的,合成的Bulk quote拷贝构造函数使用(合成的)Disc quote拷贝构造函数,后者又使用(合成的)Quote拷贝构造函数。其中,Quote拷贝构造函数拷贝bookNo和price成员;Disc_quote拷贝构造函数拷贝qty和discount成员。

值得注意的是,无论基类成员是合成的版本(如 Quote继承体系的例子)还是自定义的版本都没有太大影响。唯一的要求是相应的成员应该可访问并且不是一个被删除的函数

在我们的Quote继承体系中,所有类都使用合成的析构函数。

其中,派生类隐式地使用而基类通过将其虚析构函数定义成=default而显式地使用。

一如既往,合成的析构函数体是空的,其隐式的析构部分负责销毁类的成员。

对于派生类的析构函数来说,它除了销毁派生类自己的成员外,还负责销毁派生类的直接基类;该直接基类又销毁它自己的直接基类,以此类推直至继承链的顶端。

如前所述,Quote因为定义了析构函数而不能拥有合成的移动操作,因此当我们移动Quote对象时实际使用的是合成的拷贝操作。如我们即将看到的那样,Quote 没有移动操作意味着它的派生类也没有。

派生类中删除的拷贝控制与基类的关系

就像其他任何类的情况一样,基类或派生类也能出于同样的原因将其合成的默认构造函数或者任何一个拷贝控制成员定义成被删除的函数。

此外,某些定义基类的方式也可能导致有的派生类成员成为被删除的函数:

  1. 如果基类中的默认构造函数、拷贝构造函数、拷贝赋值运算符或析构函数是被删除的函数或者不可访问,则派生类中对应的成员将是被删除的,原因是编译器不能使用基类成员来执行派生类对象基类部分的构造、赋值或销毁操作。
  2. 如果在基类中有一个不可访问或删除掉的析构函数,则派生类中合成的默认和拷贝构造函数将是被删除的,因为编译器无法销毁派生类对象的基类部分。
  3. 和过去一样,编译器将不会合成一个删除掉的移动操作。当我们使用=default请求一个移动操作时,如果基类中的对应操作是删除的或不可访问的,那么派生类中该函数将是被删除的,原因是派生类对象的基类部分不可移动。同样,如果基类的析构函数是删除的或不可访问的,则派生类的移动构造函数也将是被删除的。

举个例子,对于下面的基类B来说:

class B
{
public:

B();
B(const B&)delete;
//其他成员,不含有移动构造函数

};

class D : public B {
//没有声明任何构造函数
D d; //正确:D的合成默认构造函数使用B的默认构造函数
D a2(d); //错误:D的合成拷贝构造函数是被删除的
d3(std::move(d));//错误:隐式地使用D的被删除的拷贝构造函数

基类B含有一个可访问的默认构造函数和一个显式删除的拷贝构造函数。

因为我们定义了拷贝构造函数,所以编译器将不会为B合成一个移动构造函数,因此,我们既不能移动也不能拷贝B的对象。

如果B的派生类希望它自己的对象能被移动和拷贝,则派生类需要自定义相应版本的构造函数。

当然,在这一过程中派生类还必须考虑如何移动或拷贝其基类部分的成员。

在实际编程过程中,如果在基类中没有默认、拷贝或移动构造函数,则一般情况下派生类也不会定义相应的操作。

移动操作与继承

如前所述,大多数基类都会定义一个虚析构函数。

因此在默认情况下,基类通常不含有合成的移动操作,而且在它的派生类中也没有合成的移动操作。

因为基类缺少移动操作会阻止派生类拥有自己的合成移动操作,所以当我们确实需要执行移动操作时应该首先在基类中进行定义

我们的Ouote可以使用合成的版本,不过前提是Quote必须显式地定义这些成员。一旦Ouote定义了自己的移动操作,那么它必须同时显式地定义拷贝操作

class Quote {
public:
Quote() = default; //对成员依次进行默认初始化
Quote(const Quote&) =default; //对成员依次拷贝
Quote (Quote&d) = default; // 对成员依次拷贝
Quotea operator=(const Quote) default; //拷贝赋值
Quotes operator=(Quote&) = default; // 移动赋值
virtual Quote() = default;
// 其他成员与之前的版本一致
};

通过上面的定义,我们就能对Quote的对象逐成员地分别进行拷贝、移动、赋值和销毁操作了。

而且除非Quote的派生类中含有排斥移动的成员,否则它将自动获得合成的移动操作。
 

派生类的拷贝控制成员

派生类构造函数在其初始化阶段中不但要初始化派生类自己的成员,还负责初始化派生类对象的基类部分.

因此,派生类的拷贝和动构造函数在拷贝和移动自有成员的同时,也要拷贝和移动基类部分的成员。类似的。派生类赋值运算符也必须为其基类部分的成员赋值。

和构造函数及赋值运算符不同的是,析构函数只负责销毁派生类自己分配的资源。

如前所述,对象的成员是被隐式销毁的:类似的,派生类对象的基类部分也是自动销毁的。

当派生类定义了拷贝或移动操作时,该操作负责拷贝或移动包括基类部分成员在内的整个对象。

定义派生类的拷贝或移动构造函数

当为派生类定义拷贝或移动构造函数时,我们通常使用对应的基类构造函数初始化对象的基类部分:

class Base {};

class D: public Base 
{public: // 默认情况下,基类的默认构造函数初始化对象的基类部分
// 要想使用拷贝或移动构造函数,我们必须在构造函数初始值列表中
// 显式地调用该构造函数
D(const D& d): Base(d) // 拷贝基类成员
/*D的成员的初始值*/
{/*...*/}

D(D&& d): Base (std::move (d)) //移动基类成员
/* D的成员的初始值*/
{}
};

初始值Base(d)将一个D对象传递给基类构造函数。

尽管从道理上来说,Base可以包含一个参数类型为D的构造函数,但是在实际编程过程中通常不会这么做。

相反,Base(d)般会匹配Base的拷贝构造函数。D类型的对象d将被绑定到该构造函数的Base&形参上。Base的拷贝构造函数负责将d的基类部分拷贝给要创建的对象。

假如我们没有提供基类的初始值的话:

//D的这个拷贝构造函数很可能是不正确的定义
// 基类部分被默认初始化,而非拷贝
D(const D& d) /*成员初始值,但是没有提供基类初始值*/
{/*...*/}

在上面的例子中,Base的默认构造函数将被用来初始化D对象的基类部分。

假定D的构造函数从a中拷贝了派生类成员,则这个新构建的对象的配置将非常奇怪:它的Base成员被赋予了默认值,而D成员的值则是从其他对象拷贝得来的。

在默认情况下,基类默认构造函数初始化派生类对象的基类部分。如果我们想拷贝(或移动)基类部分,则必须在派生类的构造函数初始值列表中显式地使用基类的拷贝(或移动)构造函数。

派生类赋值运算符

与拷贝和移动构造函数一样,派生类的赋值运算符也必须显式地为其基类部分赋值:

// Base::operator=(const Base&)不会被自动调用

D& D::operator=(conat D &rhs)
{
Base::operator=(rhs);//为基类部分赋值
// 按照过去的方式为派生类的成员赋值
// 酌情处理自赋值及释放已有资源等情况
return *this;
}

上面的运算符首先显式地调用基类赋值运算符,令其为派生类对象的基类部分赋值。

基类的运算符(应该可以)正确地处理自赋值的情况,如果赋值命令是正确的,则基类运算符将释放掉其左侧运算对象的基类部分的旧值,然后利用rhs为其赋一个新值。随后,我们继续进行其他为派生类成员赋值的工作。

值得注意的是,无论基类的构造函数或赋值运算符是自定义的版本还是合成的版本,派生类的对应操作都能使用它们。

例如,对于Base::operator=的调用语句将执行Base的拷贝赋值运算符,至于该运算符是由Base显式定义的还是由编译器合成的无关紧要。

派生类析构函数

如前所述,在析构函数体执行完成后,对象的成员会被隐式销毁。类似的,对象的基类部分也是隐式销毁的。

因此,和构造函数及赋值运算符不同的是,派生类析构函数只负责销毁由派生类自己分配的资源:

class D: public Base {
public:
// Base::~Base被自动调用执行
~D(){/*该处由用户定义清除派生类成员的操作*/}
};


对象销毁的顺序正好与其创建的顺序相反:派生类析构函数首先执行,然后是基类的析构函数,以此类推,沿着继承体反方向直至最后。

在构造函数和析构函数中调用虚函数

如我们所知,派生类对象的基类部分将首先被构建。当执行基类的构造函数时,该对象的派生类部分是未被初始化的状态。类似的,销毁派生类对象的次序正好相反,因此当执行基类的析构函数时,派生类部分已经被销毁掉了。由此可知,当我们执行上述基类成员的时候,该对象处于未完成的状态。

为了能够正确地处理这种未完成状态,编译器认为对象的类型在构造或析构的过程中仿佛发生了改变一样。

也就是说,当我们构建一个对象时,需要把对象的类和构造函数的类看作是同一个;对虚函数的调用绑定正好符合这种把对象的类和构造函数的类看成同一个的要求;对于析构函数也是同样的道理。上述的绑定不但对直接调用虚函数有效,对间接调用也是有效的,这里的间接调用是指通过构造函数(或析构函数)调用另一个函数。

为了理解上述行为,不妨考虑当基类构造函数调用虚函数的派生类版本时会发生什么情况。这个虚函数可能会访问派生类的成员,毕竟,如果它不需要访问派生类成员的话,则派生类直接使用基类的虚函数版本就可以了。然而,当执行基类构造函数时,它要用到的派生类成员尚未初始化,如果我们允许这样的访问,则程序很可能会崩溃。

如果构造函数或析构函数调用了某个虚函数,则我们应该执行与构造函数或析构函数所属类型相对应的虚函数版本。

继承的构造函数

在C++11新标准中,派生类能够重用其直接基类定义的构造函数,尽管如我们所知,这些构造函数并非是以常规的方式继承而来,但是为了方便,我们不妨姑且称其为“继承”的。

一个类只初始化它的直接基类,出于同样的原因,一个类也只继承其直接基类的构造函数。

类不能继承默认,拷贝和移动构造函数。如果派生类没有直接定义这些构造函数。则编译器将为派生类合成它们。

派生类继承基类构造函数的方式是提供一条注明了(直接)基类名的using 声明语句。

举个例子,我们可以事新定义Bulk_quote类,令其继承Disc_quote类的构造函数:

class Disc_quote
{
public:
int a;
Disc_quote(int a_){a=a_;}
};


class Bulk_quote : public Disc_quote{

public:
using Disc_quote::Disc_quote;// 继承Disc_quote的构造函数
double net_price(std::size_t) const;
};

通常情况下,using声明语句只是令某个名字在当前作用域内可见。

而当作用于构造函数时,using声明语句将令编译器产生代码。

对于基类的每个构造函数,编译器都生成一个与之对应的派生类构造函数。

换句话说,对于基类的每个构造函数,编译器都在派生类中生成一个形参列表完全相同的构造函数。
这些编译器生成的构造函数形如;

derived (parms) : base (args) ( }

其中,derived是派生类的名字,base是基类的名字,parms是构造函数的形参列表,args将派生类构造函数的形参传递给基类的构造函数。

在我们的Bulk_quote类中,继承的构造函数等价于:

Bulk_quote(int a_):Disc_quote(a_){}

如果派生类含有自己的数据成员,则这些成员将被默认初始化

继承的构造函数的特点

和普通成员的using声明不一样,一个构造函数的using声明不会改变该构造函数的访问级别。

例如,不管using声明出现在哪儿,基类的私有构造函数在派生类中还是个私有构造函数;受保护的构造函数和公有构造函数也是同样的规则。

而且,一个using 声明语句不能指定explicit或constexpr。

如果基类的构造函数是explicit或者constexpr,则继承的构造函数也拥有相同的属性。

当一个基类构透函数含有默认实参时,这些实参并不会被继承。相反,派生类将获得多个继承的构造函数,其中每个构造函数分别省略掉一个含有默认实参的形参。

例如,如果基类有一个接受两个形参的构造函数,其中第二个形参含有默认实参,则派生类将获得两个构造函数:一个构造函数接受两个形参(没有默认实参),另一个构造函数只接受一个形参,它对应于基类中最左侧的没有默认值的那个形参。

如果基类含有几个构造函数,则除了两个例外情况,大多数时候派生类会继承所有这些构造函数。

  1. 第一个例外是派生类可以继承一部分构造函数,而为其他构造函数定义自己的版本。如果派生类定义的构造函数与基类的构造函数具有相同的参数列表,则该构造函数将不会被继承。定义在派生类中的构造函数将替换继承而来的构造函数。
  2. 第二个例外是默认、拷贝和移动构造函数不会被继承。这些构造函数按照正常规则被合成。继承的构造函数不会被作为用户定义的构造函数来使用,因此,如果一个类只含有继承的构造函数,则它也将拥有一个合成的默认构造函数。
     

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1556541.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【感悟《剑指offer》典型编程题的极练之路】02字符串篇!

​ 个人主页&#xff1a;秋风起&#xff0c;再归来~ 文章所属专栏&#xff1a;《剑指offer》典型编程题的极练之路 ​​​​​​ 个人格言&#xff1a;悟已往之不谏&#xff0c;知来者犹可追 克心守己&#xff0c…

(学习日记)2024.03.25:UCOSIII第二十二节:系统启动流程详解

写在前面&#xff1a; 由于时间的不足与学习的碎片化&#xff0c;写博客变得有些奢侈。 但是对于记录学习&#xff08;忘了以后能快速复习&#xff09;的渴望一天天变得强烈。 既然如此 不如以天为单位&#xff0c;以时间为顺序&#xff0c;仅仅将博客当做一个知识学习的目录&a…

springboot通过threadLocal+参数解析器实现保存当前用户登录信息

首先先介绍一下threadLocal ThreadLocal 线程局部变量&#xff0c;创建一个线程变量后&#xff0c;针对这个变量可以让每个线程拥有自己的变量副本&#xff0c;每个线程是访问的自己的副本&#xff0c;与其他线程的相互独立。 大致知道threadLocal就可以了&#xff0c;然后我…

Vue中使用Vuex(超详细)基本使用方法

在vue中使用vuex&#xff0c;不同的vue版本要对应使用不同的vuex&#xff0c;在这里不做详情介绍&#xff0c;想具体了解的&#xff0c;请自行度娘或者必应一下。 在使用vuex之前&#xff0c;我们创建一个新的项目&#xff0c;这里我们使用的是vue的脚手架创建一个vue项目。 …

二维码门楼牌管理应用平台建设:智能匹配与高效管理

文章目录 前言一、二维码门楼牌管理应用平台的意义二、地址坐标校验的重要性三、对外采数据匹配校验的实现方式四、智能匹配与人工审核的结合五、二维码门楼牌管理应用平台的前景展望 前言 随着城市化进程的加速&#xff0c;门楼牌管理成为城市治理中不可或缺的一环。传统的门…

keil安装器件支持包

创建keil项目时&#xff0c;发现没有我们想要的器件支持包 这时我们可以选择在线安装 1.点击安装按钮 2.点击检查更新&#xff0c;下方有进度条&#xff0c;他会安装keil支持的所有器件支持包所以比较慢

Github profile Readme实现小游戏[github自述游戏]

Github profile Readme常用于个人主页介绍&#xff0c;将它与action自动化流程结合&#xff0c;可以实现一些小游戏 例如&#xff1a;2048、五子棋 2048实现 losehu (RUBO) GitHub 五子棋 https://github.com/losehu/losehu/tree/main 通过python/C编写可执行文件&#xf…

浏览器工作原理与实践--垃圾回收:垃圾数据是如何自动回收的

在上一篇文章中&#xff0c;我们提到了JavaScript中的数据是如何存储的&#xff0c;并通过例子分析了原始数据类型是存储在栈空间中的&#xff0c;引用类型的数据是存储在堆空间中的。通过这种分配方式&#xff0c;我们解决了数据的内存分配的问题。 不过有些数据被使用之后&am…

【Gitea的介绍】

&#x1f525;博主&#xff1a;程序员不想YY啊&#x1f525; &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家&#x1f4ab; &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 &#x1f308;希望本文对您有所裨益&#xff0c;如有…

BGP实训

BGP基础配置实训 实验拓扑 注&#xff1a;如无特别说明&#xff0c;描述中的 R1 或 SW1 对应拓扑中设备名称末尾数字为 1 的设备&#xff0c;R2 或 SW2 对应拓扑中设备名称末尾数字为2的设备&#xff0c;以此类推&#xff1b;另外&#xff0c;同一网段中&#xff0c;IP 地址的主…

基于重写ribbon负载实现灰度发布

项目结构如下 代码如下&#xff1a; pom&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocat…

淘宝商品详情数据(商品分析,竞品分析,代购商城建站与跨境电商,ERP系统商品数据选品)

淘宝商品详情数据在多个业务场景中发挥着关键作用&#xff0c;以下是一些主要的应用场景&#xff1a; 请求示例&#xff0c;API接口接入Anzexi58 商品分析&#xff1a;通过对淘宝商品详情的全面分析&#xff0c;商家可以深入了解商品的属性、价格、销售量、评价等信息。这些数…

你真的会数据结构吗:二叉树

❀❀❀ 文章由不准备秃的大伟原创 ❀❀❀ ♪♪♪ 若有转载&#xff0c;请联系博主哦~ ♪♪♪ ❤❤❤ 致力学好编程的宝藏博主&#xff0c;代码兴国&#xff01;❤❤❤ halo铁汁们&#xff0c;没错又是你们人见人爱&#xff0c;花见花开的大伟啊&#xff0c;今天也是周六&#x…

node.js学习(2)

版权声明 以下文章为尚硅谷PDF资料&#xff0c;B站视频链接&#xff1a;【尚硅谷Node.js零基础视频教程&#xff0c;nodejs新手到高手】仅供个人学习交流使用。如涉及侵权问题&#xff0c;请立即与本人联系&#xff0c;本人将积极配合删除相关内容。感谢理解和支持&#xff0c;…

椋鸟数据结构笔记#3:链表

萌新的学习笔记&#xff0c;写错了恳请斧正。 目录 链表的定义 链表的分类 方向&#xff08;单向还是双向&#xff09; 头节点&#xff08;哨兵节点&#xff09;的有无 循环或不循环 8种分类 不带头单向不循环链表的实现 带头单向循环链表的实现 链表与顺序表的差异 链…

java全排列(力扣Leetcode46)

全排列 力扣原题链接 问题描述 给定一个不含重复数字的数组 nums&#xff0c;返回其所有可能的全排列。你可以按任意顺序返回答案。 示例 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]] 示例 2…

FME学习之旅---day17

我们付出一些成本&#xff0c;时间的或者其他&#xff0c;最终总能收获一些什么。 【FME-HOW-TO系列】28 栅格邻域函数 RasterConvolver转换器说明&#xff1a; 接受包含栅格几何对象的输入要素&#xff0c;并在对所有波段应用卷积滤波 器后输出要素。 本人对栅格数据处理的较…

从易到难,推荐9个适合练手的C++项目

老有一些同学和我说学习了 C 以后&#xff0c;想要做些项目锻炼自己&#xff0c;让我从「简单到难」都推荐一些。 那有啥说的&#xff0c;必须推荐&#xff01;毕竟 C 的优质项目我见过太多了&#xff01; 下面我就按照「从易到难」的梯度&#xff0c;依次来推荐&#xff0c;…

反应式编程(一)什么是反应式编程

目录 一、背景二、反应式编程简介2.1 定义2.2 反应式编程的优势2.3 命令式编程 & 反应式编程 三、Reactor 入门3.1 Reactor 的核心类3.2 Reactor 中主要的方法1&#xff09;创建型方法2&#xff09;转化型方法3&#xff09;其他类型方法4&#xff09;举个例子 四、Reactor …

kafka学习笔记02(小滴课堂)

Kafka命令行生产者发送消息和消费者消费消息实战 已存在的kafka不能重复创建。 broker设置的是1&#xff0c;factor大于broker了&#xff0c;所以报错。 生产者发送消息&#xff1a; kafka列表出现了新的kafka。 我们使用这个kafka。 我们启动消费者&#xff1a; 我们现在不从…