【C++】继承(详解)
前言:今天我们正式的步入C++进阶内容的学习了,当然了既然是进阶意味着学习难度的不断提升,各位一起努力呐。
💖 博主CSDN主页:卫卫卫的个人主页 💞
👉 专栏分类:高质量C++学习 👈
💯代码仓库:卫卫周大胖的学习日记💫
💪关注博主和博主一起学习!一起努力!
什么是继承
C++中的继承是一种面向对象编程的特性,它允许一个类(子类)继承另一个类(父类)的属性和方法,并且可以添加自己的属性和方法。通过继承,子类可以重用父类的代码,减少重复编写代码的工作量。在C++中,使用关键字"extends"可以声明一个类继承另一个类。子类将自动继承父类的非私有成员和方法,并可以通过重写(override)父类的方法或添加新方法来实现自己的行为。(光看文字大家肯定还是觉得晦涩难懂的,我们直接看代码)
继承的使用格式
在C++中,使用继承的格式如下:
class 子类名 : 访问权限 基类名
{// 子类的成员和函数声明
};
其中,子类名表示要定义的子类的名称,访问权限可以是public、protected或private,用于指定从基类继承成员的访问权限,基类名表示要继承的基类的名称。
例如,定义一个基类(父类)father和一个派生类(子类)son:
class father//父类
{
public:string name = "张三";int age = 20;
};class son: public father//public继承子类可以直接继承父类的公有对象
{
public :void print(){cout << name << endl << age << endl;}
};
int main()
{son s1;s1.print();return 0;
}
在上述例子中son是parent的子类,使用public权限继承了parent中name和age,然后用子类中的成员函数去调用print函数从而调用了父类中的成员。
结果如下:
如果还是不太能理解父类和子类的关系,看下图,通俗的理解就是儿子继承了父亲的遗产,并且儿子的财产依然存在。
继承后子类(派生类)的访问权限
在C++中,继承后的访问权限可以分为三种:public、protected和private。
公有继承(public inheritance):当通过公有继承派生一个子类时,基类的公有成员和保护成员在子类中仍然是公有的,可以直接访问。基类的私有成员在子类中是不可访问的。
保护继承(protected inheritance):当通过保护继承派生一个子类时,基类的公有成员和保护成员在子类中都变成了保护成员,它们通过子类只能被子类自身和子类的派生类访问,外部无法访问。基类的私有成员在子类中是不可访问的.。
私有继承(private inheritance):当通过私有继承派生一个子类时,基类的公有成员和保护成员在子类中都变成了私有成员,只能在子类内部访问,外部无法访问。基类的私有成员在子类中是不可访问的。
需要注意的是,继承的访问权限仅影响继承后的成员的访问权限,不会改变基类中成员本身的访问权限。
实例演示:
class Base {
public:int publicMember;
protected:int protectedMember;
private:int privateMember;
};class PublicDerived : public Base {
public:void example() {publicMember = 10; // 可以直接访问公有成员protectedMember = 20; // 可以直接访问保护成员// privateMember = 30; 无法访问私有成员}
};class ProtectedDerived : protected Base {
public:void example() {publicMember = 10; // 可以直接访问公有成员protectedMember = 20; // 可以直接访问保护成员// privateMember = 30; 无法访问私有成员}
};class PrivateDerived : private Base {
public:void example() {publicMember = 10; // 可以直接访问公有成员protectedMember = 20; // 可以直接访问保护成员// privateMember = 30; 无法访问私有成员}
};
总结:
- 基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它。
- 基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,但需要在派生类中能访问,就定义为protected。可以看出保护成员限定符是因继承才出现的。
- 实际上面的表格我们进行一下总结会发现,基类的私有成员在子类都是不可见。基类的其他
成员在子类的访问方式 == Min(成员在基类的访问限定符,继承方式),public > protected > private。 - 使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式。
- 在实际运用中一般使用都是public继承,几乎很少使用protetced/private继承,也不提倡使用protetced/private继承,因为protetced/private继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强。
- 很多人分不清保护继承和私有继承,其实区别就是保护继承他的儿子的儿子也能访问他的保护成员和公有成员,私有继承就是只有他儿子能访问。
子类与父类
子类与父类的相互赋值转换
- 符号赋值“=”
注:这里我们先暂时知道赋值的规则就是切片赋值,就是把子类中的父类那部分,赋值过去即可。
在C++中,子类对象可以被隐式地转换为父类对象,但是父类对象不能被隐式地转换为子类对象。这是因为子类对象继承了父类对象的成员和方法,但父类对象并不包含子类特有的成员和方法。
以下是一个示例说明子类对象转换为父类对象的情况:
class Parent {
public:void print() {std::cout << "Parent class" << std::endl;std::cout << age << std::endl;}int age = 30;
};class Child : public Parent {
public:void print() {std::cout << "Child class" << std::endl;}
};int main() {Child child;child.age = 50;Parent parent = child; // 子类对象隐式转换为父类对象parent.print(); // 调用父类的print()方法return 0;
}
- 引用
Child s1;//子类Parent& t1=st;//父类,通过引用对其进行赋值
- 指针
Child s1;//子类Parent* t1 = &st;//父类,通过指针进行赋值
总结来说,C++中可以将子类对象隐式地转换为父类对象,但是父类对象无法隐式地转换为子类对象。如果需要使用子类特有的成员和方法,需要进行类型转换.
同名成员变量
在C++中,当子类继承了父类时,如果子类中出现了与父类同名的成员变量,那么子类的同名成员变量会隐藏父类的同名成员变量。这称为"隐藏"(hiding)。
具体来说,如果子类定义了与父类同名的成员变量,那么子类中的同名成员变量会隐藏父类中的同名成员变量。这意味着当通过子类对象访问同名成员变量时,会优先访问子类中的成员变量,而无法直接访问到父类中的同名成员变量。
#include <iostream>
class Parent {
public:int num = 10;
};class Child : public Parent {
public:int num = 20;
};int main() {Child child;std::cout << "Child num: " << child.num << std::endl; // 编译器还是就近原则,优先访问子类中的num,输出: Child num: 20std::cout << "Parent num: " << child.Parent::num << std::endl; //想要访问父类中的num,需要指定类域,输出: Parent num: 10return 0;
}
在上面的示例中,Parent类中定义了一个名为num的成员变量,初始值为10。Child类继承了Parent类,并定义了一个同名成员变量num,初始值为20。
在main函数中,我们创建了Child类的对象child。当我们通过子类对象child访问num时,会优先访问子类的同名成员变量,因此输出结果为20。如果我们需要访问父类中的同名成员变量,可以使用作用域解析符"::"来指定父类,如child.Parent::num
同名成员函数
在C++中,当子类继承了父类时,如果子类中出现了与父类同名的成员函数,那么子类的同名成员函数将隐藏父类的同名成员函数。
具体来说,如果子类定义了与父类同名的成员函数,那么子类中的同名成员函数将隐藏父类中的同名成员函数。这意味着当通过子类对象调用同名成员函数时,会优先调用子类中的成员函数,而无法直接调用到父类中的同名成员函数。
然而,与成员变量的隐藏不同的是,对于成员函数的隐藏,如果需要在子类中使用父类的同名成员函数,可以使用作用域解析符"::"来指定父类.
class Parent {
public:void print() {std::cout << "Parent print" << std::endl;}
};class Child : public Parent {
public:void print() {std::cout << "Child print" << std::endl;Parent::print();//可以通过指定域来调用}
};int main() {Child child;child.print(); // 输出: Child printchild.Parent::print(); // 输出: Parent printreturn 0;
}
子类中的默认成员函数
构造函数
在C++中,子类可以通过构造函数来初始化继承的父类成员。子类的构造函数可以调用父类的构造函数来初始化父类的成员变量。子类的构造函数在构造子类对象时被调用,而且会在子类构造函数的初始化列表中调用父类的构造函数.
注: 编译器会默认先调用父类的构造函数,再调用子类的构造函数.
class Parent
{
public:Parent(string name = "Dad", int age = 30):_name(name),_age(age){cout << "这是父类:" << _name << " " << age << endl;}protected:string _name;int _age;
};class Child: public Parent
{
public:Child(string son):_son(son){cout << "这是子类:" << _son << endl;}protected:string _son;
};int main()
{Child s1("son");return 0;
}
这里我们可以明显的知道,编译器先调用父类的默认构造,在调用子类的构造,所以一定要保证这里的父类的构造是有效的,以防止构造失效.
析构函数
在C++中,子类可以定义自己的析构函数,用来释放子类对象在内存中分配的资源。子类的析构函数与基类的析构函数类似,通过在类的声明中使用特殊的名称~类名()来定义。
子类的析构函数会自动调用基类的析构函数,以确保基类对象中分配的资源被正确释放。
注: 析构函数和构造函数相反,编译器默认先调用子类的析构函数,再调用父类的析构函数.
实例演示:
class Parent
{
public:~Parent(){cout << "这是父类的析构" << endl;}
protected:string _name;int _age;
};class Child: public Parent
{
public:~Child(){cout << "这是子类的析构" << endl;}
protected:string _son;
};int main()
{Child s1;return 0;
}
拷贝构造
在C++中,子类可以定义自己的拷贝构造函数,用于创建一个新的子类对象,该对象与已存在的子类对象具有相同的值。
注: 子类的拷贝构造函数会自动调用基类的拷贝构造函数,以确保基类对象的成员变量得到正确的复制.
class Parent
{
public:Parent(string name = "Dad", int age = 30):_name(name), _age(age){cout << "这是父类:" << _name << " " << age << endl;}protected:string _name;int _age;
};class Child : public Parent
{
public:Child(string son):_son(son){cout << "这是子类:" << _son << endl;}Child(Child& child):Parent(child)//通过切片的方式直接对父类进行了初始化,_son(child._son){cout << "拷贝构造:" << _name << " " << _age << " " << _son << endl;}
protected:string _son;
};int main()
{Child s1("son");Child s2(s1);return 0;
}
赋值运算符重载
子类的赋值运算符重载函数可以用来实现对象之间的赋值操作,包括赋值基类对象的成员变量和子类对象的成员变量。在赋值过程中,可以先调用基类的赋值运算符重载函数来赋值基类对象的成员变量,然后再进行子类对象的成员变量的赋值操作.
class Parent
{
public:Parent(string name = "Dad", int age = 30):_name(name), _age(age){cout << "这是父类:" << _name << " " << age << endl;}Parent& operator=(const Parent& s1){if (this != &s1){cout << "调用父类" << endl;_name = s1._name;_age = s1._age;}}protected:string _name;int _age;
};class Child : public Parent
{
public:Child(string son):_son(son){cout << "这是子类:" << _son << endl;}Child(Child& child):Parent(child),_son(child._son){cout << "拷贝构造:" << _name << " " << _age << " " << _son << endl;}Child& operator=(const Child& s1){if (this != &s1){Parent::operator=(s1);//调用父类的运算符重载,以免调用自身造成栈溢出_son = s1._son;}}
protected:string _son;
};int main()
{Child s1("son");Child s2(s1);return 0;
}
继承中的单继承与多继承
在C++中,单继承和多继承是两种不同的继承方式。
-
单继承:
单继承指的是一个派生类只能继承自一个基类(一个孩子一个父亲)。在单继承中,一个派生类可以继承基类的所有成员(包括成员函数和成员变量),并且可以通过派生类对象访问这些成员。单继承的语法如下:class Base {// base class members };class Derived : public Base {// derived class members };
在这个例子中,
Derived
类从Base
类单继承,Derived
类可以访问Base
类中的所有公有成员。 -
多继承:
多继承指的是一个派生类能够从多个基类继承。在多继承中,一个派生类可以继承多个基类的成员,这些成员可以通过派生类对象访问(一个孩子有多个父亲)。多继承的语法如下:class Base1 {// base class 1 members };class Base2 {// base class 2 members };class Derived : public Base1, public Base2 {// derived class members };
在这个例子中,
Derived
类从Base1
类和Base2
类多继承,Derived
类可以访问Base1
和Base2
类中的所有公有成员。
需要注意的是,多继承在设计上可能会引入复杂性和冲突,需要谨慎使用。在多继承中,如果多个基类具有相同的成员函数或成员变量,派生类必须显式指明使用哪个基类的成员。另外,多继承还可能导致菱形继承问题,即继承图中出现多个路径指向同一个基类。为了解决这个问题,可以使用虚继承或其他技术。
总结起来,单继承和多继承是C++中的两种继承方式。单继承指的是派生类只能继承自一个基类,而多继承则允许派生类从多个基类继承。使用继承时需要考虑设计的简洁性和灵活性,以及可能引入的复杂性和冲突。
菱形继承
C++中的菱形继承(diamond inheritance)是多继承的一个特殊情况。当一个派生类通过两个不同的路径继承自同一个基类时,就会形成菱形继承结构。这种继承结构的名称源于它的图形表示类似于菱形。
考虑以下示例代码:
class A {
public:void funcA() {cout << "A::funcA()" << endl;}
};class B : public A {
public:void funcB() {cout << "B::funcB()" << endl;}
};class C : public A {
public:void funcC() {cout << "C::funcC()" << endl;}
};class D : public B, public C {
public:void funcD() {cout << "D::funcD()" << endl;}
};
在这个例子中,类D
通过B
和C
两个路径分别继承了类A
,形成了菱形继承结构。这意味着类D
会有两份来自A
类的成员函数和成员变量。例如,D
类实例可以调用funcA()
方法两次,一次是通过B
的路径,另一次是通过C
的路径。
菱形继承可能导致以下问题:
-
虚函数二义性(Virtual Function Ambiguity):如果在菱形继承结构中,派生类中存在同名的虚函数,那么在派生类中访问这个虚函数时将产生二义性,编译器不知道应该使用哪个父类的虚函数实现。
-
数据冗余(Data Redundancy):由于菱形继承结构中派生类从两个不同的路径继承了同一个基类,导致派生类中会存在两份相同的成员变量。
为了解决菱形继承带来的问题,可以使用以下方法:
- 虚拟继承(Virtual Inheritance)是C++中一种特殊的继承方式,用于解决多继承中的菱形继承问题。通过使用虚拟继承,可以确保在派生类中只包含一个共享的基类子对象,避免了数据冗余和虚函数二义性的问题。
虚拟继承的特点有:
虚拟继承只发生在最顶层的派生类中,而不会在派生类的派生类中发生。也就是说,只有直接派生自基类的派生类才能使用虚拟继承。
虚拟继承会在内存布局中引入一个额外的指针(虚表指针)来维护虚拟继承的关系。这个指针指向了虚拟基类的子对象,用于访问虚基类的成员。
如果虚拟基类有自己的派生类,那么虚拟继承会优先选择这个派生类中的虚基类子对象作为共享对象
- 重写和调用具体的父类函数:通过在派生类中重写同名的虚函数,并在派生类中显式调用具体的父类函数,可以解决虚函数二义性的问题。
综上所述,菱形继承是C++多继承的一个特殊情况,可能会引发虚函数二义性和数据冗余的问题,可以通过虚继承和重写父类函数的方式来解决这些问题。
好啦,今天的内容就到这里啦,下期内容预告C++中的多态.
结语:进阶的内容有点繁杂,大家一起加油呐!。
相关文章:

【C++】继承(详解)
前言:今天我们正式的步入C进阶内容的学习了,当然了既然是进阶意味着学习难度的不断提升,各位一起努力呐。 💖 博主CSDN主页:卫卫卫的个人主页 💞 👉 专栏分类:高质量C学习 👈 &#…...

网络io与select,poll,epoll
前言 网络 IO,会涉及到两个系统对象,一个是用户空间调用 IO 的进程或者线程,另一个是内核空间的内核系统,比如发生 IO 操作 read 时,它会经历两个阶段: 1. 等待数据准备就绪 2. 将数据从内核拷贝到进程或…...

【Linux】多线程(一万六千字)
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 文章目录 前言 线程的概念 线程的理解(Linux系统为例) 在Linux系统里如何保证让正文部分的代码可以并发的去跑呢? 为什么要有多进程呢? 为…...

sh脚本笔记2
test条件测试 语法 条件测试语法说明语法1:test <测试表达式>这是利用test命令进行条件测试表达式的方法。test命令和“<测试表达式>”之间至少有一个空格语法2:[ <测试表达式> ]这是通过[](单中括号)进行条件…...

js替换对象里面的对象名称
data为数组,val为修改前的名称,name为修改后的名称 JSON.parse(JSON.stringify(data).replace(/val/g, name)) ; 1.替换data里面的对象tenantInfoRespVO名称替换成tenantInfoUpdateReqVO 2.替换语句: 代码可复制 let tenantInf…...

鸿蒙开发设备管理:【@ohos.settings (设置数据项名称)】
设置数据项名称 说明: 本模块首批接口从API version 8开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。 本模块提供设置数据项的访问功能相关接口的说明及示例。 导入模块 import settings from ohos.settings;settings.getUri…...

STM32之五:TIM定时器(2-通用定时器)
目录 通用定时器(TIM2~5)框图 1、 输入时钟源选择 2、 时基单元 3 、输入捕获:(IC—Input Capture) 3.1 输入捕获通道框图(TI1为例) 3.1.1 滤波器: 3.1.2 边沿检测器…...

【分布式系统】监控平台Zabbix对接grafana
以前两篇博客为基础 【分布式系统】监控平台Zabbix介绍与部署(命令截图版)-CSDN博客 【分布式系统】监控平台Zabbix自定义模版配置-CSDN博客 一.安装grafana并启动 添加一台服务器192.168.80.104 初始化操作 systemctl disable --now firewalld set…...

操作系统真象还原:编写硬盘驱动程序
第13章-编写硬盘驱动程序 这是一个网站有所有小节的代码实现,同时也包含了Bochs等文件 13.1 硬盘及分区表 13.1.1 创建从盘及获取安装的磁盘数 要实现文件系统,必须先有个磁盘介质,虽然咱们己经有个虚拟磁盘 hd60M.img,但它只…...

firewalld防火墙(二)
一:firewalld高级配置 1:关于iptables的知识 iptables 是Linux系统中传统的命令行防火墙管理工具,它基于内核的netfilter框架工作,用于配置和管理网络规则集,比如过滤(允许/拒绝)进出的数据包…...

Android-悬浮窗口
在Android系统中,如果应用需要弹出一个悬浮窗口,就需要申请一项特殊权限 <uses-permission android:name"android.permission.SYSTEM_ALERT_WINDOW"/>在Android O之前的系统中申请了该权限后,再给对应的window设置 WindowM…...

打破僵局:Foxit Reader无法打开的终极解决方案
打破僵局:Foxit Reader无法打开的终极解决方案 在数字化阅读时代,Foxit Reader作为一款广受欢迎的PDF阅读器,其打不开的问题无疑会给用户带来诸多不便。本文将为您提供全面的解决方案,从基础检查到高级技巧,确保您能够…...

[调试] JTAG下运行正常,从QSPI或者SD卡启动则无响应,如何查找问题
[调试] JTAG下运行正常,从QSPI或者SD卡启动则无响应,如何查找问题 一、问题现象二、用自定义fsbl替代系统默认的fsbl1. 新建fsbl_new2. 如果提示缺少xilffs库3. 使能调试信息输出 三. 启动成功和失败情况下的典型输出1. JTAG启动模式: 正常加载2. QSPI启…...

Linux内核 -- 多线程之wait_event用法
Linux Kernel 中 wait_event 的高级用法及注意事项 在Linux内核编程中,wait_event 系列函数是用于实现进程等待和事件通知机制的重要工具。本文将详细介绍 wait_event 的高级用法以及注意事项。 1. 基本用法 wait_event 系列宏主要包括以下几种形式: …...

双指针系列第 8 篇:盛水最多的容器。几句话讲明白!
Leetcode 题目链接 思路 取首尾双指针和水量如下所示,设高度函数为 h ( i ) h(i) h(i),在下图中 h ( l ) < h ( r ) h(l) < h(r) h(l)<h(r)。 观察以 l l l 为左边界所能构成的其他水量,与矮的右边界搭配结果如下。 与高的…...

c++高阶-1-模板
文章目录 模板一、模板基本语法二、函数模板1.基本语法2.函数模板注意事项3.普通函数和函数模板区别4.普通函数和函数模板调用规则 三、类模板1.基本语法2.类模板和函数模板的区别3.类模板中成员函数调用时机4.类模板对象做函数参数5.类模板与继承6.成员函数的类外实现 模板 一…...

.net core 的 winform 的 浏览器控件 WebView2
在.NET Core WinForms应用程序中,没有直接的“浏览器控件”,因为WinForms不支持像WebBrowser控件那样的功能。但是,你可以使用WebView2控件,它是一个基于Chromium的浏览器内核,可以在WinForms应用程序中嵌入Web内容。 …...

Django QuerySet对象,all()方法
all()方法 在Django中,all()方法是QuerySet对象的一个方法,用于获取模型的所有实例。 当你调用ModelName.objects.all()时,Django会生成一个SQL查询,从数据库中获取该模型的所有记录,并返回一个QuerySet对象…...

自动生成网站sitemap
要在 Next.js 和 Contentlayer 项目中实现自动生成 Sitemap 的功能,你可以编写一个脚本,在每次生成文档后自动生成 Sitemap。以下是一个示例脚本,你可以根据自己的需求进行调整。 步骤 1:安装必要的依赖 首先,你需要…...

中国经济昆虫志(55卷)
中国经济昆虫志,共55卷,内容包括概述、形态特征、分类等。各级分类单元均编有检索表,每个种有特征描述、地理分布,有的还记载有生活习性和防治方法。为便于鉴定,绘制有特征图和彩色图。 包括鞘翅目天牛科、半翅目蝽科、…...

linux环境安装elasticsearch缓存数据库和Kibana客户端
linux环境安装elasticsearch缓存数据库,今天我们安装7.17.18版本,并分析遇到的问题。 一、elasticsearch安装运行 1、直接下载 wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-7.17.18-linux-x86_64.tar.gz2、解压 tar -…...

OpenSSL的一些使用案例
目录 一、介绍 二、基本使用 1、Shell (1)文件加解密 (2)生成密钥文件 2、API (1)md5sum (2)AES256加解密 一、介绍 本篇博客重点不是详细描述 OpenSSL 的用法,只…...

常用字符串方法<python>
导言 在python中内置了许多的字符串方法,使用字符串方法可以方便快捷解决很多问题,所以本文将要介绍一些常用的字符串方法。 目录 导言 string.center(width[,fillchar]) string.capitalize() string.count(sub[,start[,end]]) string.join(iterabl…...

线程池666666
1. 作用 线程池内部维护了多个工作线程,每个工作线程都会去任务队列中拿取任务并执行,当执行完一个任务后不是马上销毁,而是继续保留执行其它任务。显然,线程池提高了多线程的复用率,减少了创建和销毁线程的时间。 2…...

Python28-5 k-means算法
k-means 算法介绍 k-means 算法是一种经典的聚类算法,其目的是将数据集分成 ( k ) 个不同的簇,每个簇内的数据点尽可能接近。算法的基本思想是通过反复迭代优化簇中心的位置,使得每个簇内的点与簇中心的距离之和最小。k-means 算法的具体步骤…...

主流国产服务器操作系统技术分析
主流国产服务器操作系统 信创 "信创",即信息技术应用创新,作为科技自立自强的核心词汇,在我国信息化建设的进程中扮演着至关重要的角色。自2016年起步,2020年开始蓬勃兴起,信创的浪潮正席卷整个信息与通信技…...

【Linux】线程封装与互斥(万字)
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 文章目录 前言 C多线程的用法 对原生线程进行一次封装 理解pthread线程 Linux线程互斥 进程线程间的互斥相关背景概念 互斥量mutex 操作共享变量会有问题的售票…...

5分钟教你部署MySQL8.0环境
此方法基于Windows操作系统! 一、在MySQL官网单击downloads(下载)MySQLhttps://www.mysql.com/cn/ 选择在Windows操作系统下载 二、选择合适的版本 推荐下载第二种,安装时离线安装即可 三、安装MySQL8.0 1、找到MySQL下载完成…...

LLM应用:传统NLP任务
LLM出来以后,知乎上就出现了“传统NLP已死”的言论,但是传统NLP真的就被扔进历史的垃圾桶了吗? 其实,尽管LLM具有出色的通用能力,但仍然无法有效应对低资源领域的自然语言处理任务,如小语种翻译。为了更好地…...

基于Hadoop平台的电信客服数据的处理与分析③项目开发:搭建Kafka大数据运算环境---任务11:基础环境准备
任务描述 任务主要是安装配置基础环境,主要内容包括: 1、安装java Kafka和ZooKeeper都需要安装Java环境,推荐至少Java8及以上版本 2、安装ZooKeeper ZooKeeper是Kafka集群的必要组件 3、安装kafka Kafka版本包括使用的scala语言版本和kafka版…...