详解C++类和对象(中(类的6个默认成员函数))

文章目录

  • 写在前面
  • 1. 类的6个默认成员函数
  • 2. 构造函数
    • 2.1 构造函数的引入
    • 2.1 构造函数的特性
  • 3. 析构函数
    • 3.1 析构函数的引入
    • 3.2 析构函数的特性
  • 4. 拷贝构造函数
    • 4.1 拷贝构造函数概念
    • 4.2 拷贝构造函数的特性
    • 4.3 拷贝构造函数典型调用场景
  • 5. 赋值运算符重载
    • 5.1 运算符重载
    • 5.2 赋值运算符重载
  • 6. const成员函数
  • 7. 取地址及const取地址操作符重载

写在前面

这篇文章详细介绍了类的 6 个默认成员函数,它们是构造函数、析构函数、拷贝构造函数、赋值运算符重载、取地址和 const 取地址操作符重载以及const 成员函数。这些成员函数在 C++ 中是默认生成的,默认成员函数在类的设计和实现中起着非常重要的作用,下面我们来一一介绍。

1. 类的6个默认成员函数

如果一个类中什么成员都没有,简称为空类。
空类中真的什么都没有吗?并不是,在 C++ 中,即使一个类中看起来什么都没有,编译器会自动生成以下6个默认成员函数
默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数
在这里插入图片描述

2. 构造函数

2.1 构造函数的引入

例如有如下一个类:

#include <iostream>
using namespace std;class Date
{
public:void Init(int year, int month, int day){_year = year;_month = month;_day = day;}void Print(){cout << _year << '/' << _month << '/' << _day << endl;}private:int _year;int _month;int _day;
};int main()
{Date d1, d2;d1.Init(2024, 2, 5);d2.Init(2024, 2, 6);d1.Print();d2.Print();return 0;
}

对于上面的Date类,可以通过 Init 公有方法给对象设置日期,但如果每次创建对象时都调用该方法设置信息,未免有点麻烦,那能否在对象创建时,就将信息设置进去呢?

在C++中为了让对象在实例化的时候能够完成初始化,提供了构造函数。构造函数是一种特殊的成员函数,用于在创建对象时对其进行初始化。在 C++ 中,构造函数的名称与类名相同,不返回任何值,甚至没有 void 类型的返回值。构造函数的主要作用是初始化对象的数据成员,确保对象在创建时有 一个合适的初始值。创建类类型对象时由编译器自动调用,并且在对象整个生命周期内只调用一次

2.1 构造函数的特性

  1. 函数名与类名相同。
  2. 无返回值。
  3. 对象实例化时编译器自动调用对应的构造函数。
    在这里插入图片描述
  4. 构造函数可以重载。
    在这里插入图片描述
  5. 如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦
    用户显式定义编译器将不再生成。
    在这里插入图片描述
  6. 关于编译器生成的默认成员函数,我们会有疑惑:不实现构造函数的情况下,编译器会
    生成默认的构造函数。但是看起来默认构造函数又没什么用?d对象调用了编译器生成的默
    认构造函数,但是d对象_year/_month/_day,依旧是随机值。也就说在这里编译器生成的
    默认构造函数并没有什么用??

    其实不然,这是因为C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型,看看下面的程序,就会发现编译器生成默认的构造函数会对自定类型成员_t调用的它的默认成员
    函数。
    在这里插入图片描述
    注意:C++11 中针对内置类型成员不初始化的缺陷,又打了补丁,即:内置类型成员变量在
    类中声明时可以给默认值。

    在这里插入图片描述
  7. 无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。
    注意:无参构造函数全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数即不需要传参就可以调用的构造函数
    在这里插入图片描述
    在实际编程中,如果对象的初始化值不依赖于外部参数,并且没有特殊的初始化需求,编译器生成的默认构造函数就够用。如果对象的初始化值依赖于外部参数,推荐缺省的默认构造函数,因为全缺省的传不传参都可以调用。

3. 析构函数

3.1 析构函数的引入

例如有如下一个类:

#include <iostream>
using namespace std;class Stack
{public:void Init(int capacity = 4){int* tmp = (int*)malloc(sizeof(int) * capacity);if (tmp == nullptr){perror("malloc fail");exit(-1);}_nums = tmp;_capacity = capacity;_top = 0;}void Destroy(){if (_nums){free(_nums);_nums = nullptr;_capacity = _top = 0;}}void push(int x){//检查扩容//..._nums[_top++] = x;}
private:int* _nums;int _top;int _capacity;};
int main()
{Stack st;st.Init();st.push(1);st.push(1);st.push(1);st.Destroy();return 0;
}

对于上面的Stack类,可以通过Destroy公有方法清理对象中的资源(动态申请的空间),但如果每次使用完对象时都调用该方法来主动的释放资源,未免有点麻烦,那能否在对象销毁之前,就自动调用相关函数来清理对象中的资源呢?

在C++中,可以通过析构函数来实现在对象销毁之前自动清理对象中的资源。析构函数是特殊的成员函数,该函数与构造函数功能相反,完成的不是对象的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会编译器会自动调用析构函数,完成对象中资源的清理工作

3.2 析构函数的特性

  1. 析构函数名是在类名前加上字符 ~。

  2. 无参数无返回值类型。

  3. 对象生命周期结束时,C++编译系统系统自动调用析构函数。
    在这里插入图片描述

  4. 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载。

  5. 关于编译器自动生成的析构函数,是否会完成一些事情呢?下面的程序我们会看到,编译器
    生成的默认析构函数,对自定类型成员调用它的析构函数。
    在这里插入图片描述

#include <iostream>
using namespace std;
class B
{
public:B(int b = 0){_b = b;cout << "B(int b = 0)" << endl;}~B(){cout << "~B()" << endl;}
private:int _b;
};class A
{
public:private:int _a;B _bb;
};int main()
{A a;return 0;
}

上面程序运行结束后输出:~B()。
而在main方法中根本没有直接创建B类的对象,为什么最后会调用B类的析构函数?
因为:main函数中创建了A类的对象a,而a中包含2个成员变量,其中_a是内置类型成员,销毁时不需要资源清理,最后系统直接将其内存回收即可;而_bb是B类对象,所以在a销毁时,要将其内部包含的B类的_bb对象销毁,所以要调用B类的析构函数。但是:main函数中不能直接调用B类的析构函数,实际要释放的是A类对象,所以编译器会调用A类的析构函
数,而A没有显式提供,则编译器会给A类生成一个默认的析构函数,目的是在其内部调用B类的析构函数,即当A对象销毁时,要保证其内部每个自定义对象都可以正确销毁,main函数中并没有直接调用B类析构函数,而是显式调用编译器为A类生成的默认析构函数。
注意:创建哪个类的对象则调用该类的构造函数,销毁那个类的对象则调用该类的析构函数。

  1. 如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数,比如Date类;有资源申请时,一定要写,否则会造成资源泄漏,比如Stack类。
    在这里插入图片描述

4. 拷贝构造函数

4.1 拷贝构造函数概念

拷贝构造函数是C++中的一种特殊成员函数,它通常用于在对象创建时,使用一个现有对象的内容来初始化新对象,从而实现对象的拷贝。
拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用
语法形式如下:

class Date
{
public://默认构造函数Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}//拷贝构造函数,使用同类类型的对象初始化Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}
private:int _year;int _month;int _day;
};int main()
{Date d1(2024, 2, 6);Date d2(d1);//使用 d1 拷贝构造 d2return 0;
}

4.2 拷贝构造函数的特性

  1. 拷贝构造函数是构造函数的一个重载形式。
    在这里插入图片描述
    需要注意的是:显示的写了拷贝构造以后,编译器就不会生成拷贝构造函数了,同时默认构造函数,编译器不会生成了,因为拷贝构造函数也是构造函数。
    在这里插入图片描述

  2. 拷贝构造函数的参数只有一个必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用
    在这里插入图片描述

  3. 若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按
    字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。
    在这里插入图片描述

  4. 编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗?
    当然像上面这种类是没必要的。那么下面的类呢?

#include <iostream>
using namespace std;class Stack
{public://构造函数Stack(int capacity = 4){int* tmp = (int*)malloc(sizeof(int) * capacity);if (tmp == nullptr){perror("malloc fail");exit(-1);}_nums = tmp;_capacity = capacity;_top = 0;}//析构函数~Stack(){cout << "~Stack()" << endl;if (_nums){free(_nums);_nums = nullptr;_capacity = _top = 0;}}private:int* _nums;int _top;int _capacity;
};
int main()
{Stack st1;Stack st2(st1);//st1拷贝构造st2return 0;
}

当我们运行上面的程序时,发现代码崩溃,下面我们来分析一下,为什么代码会崩溃呢?
在这里插入图片描述
因此我们得出结论:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝。
在这里插入图片描述

4.3 拷贝构造函数典型调用场景

现有如下一个Date类:

#include <iostream>
using namespace std;
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){cout << "Date(const Date& d)" << endl;_year = d._year;_month = d._month;_day = d._day;}private:int _year;int _month;int _day;
};
  1. 使用已存在对象创建新对象
int main()
{Date d1;Date d2(d1);//使用 d1 拷贝构造 d2Date d3 = d2;//使用 d2 拷贝构造 d3return 0;
}

运行结果如下:
在这里插入图片描述
2. 函数参数类型为类类型对象

void func(Date d)
{cout << "void func(Date d)" << endl;
}
int main()
{Date d1;func(d1);return 0;
}

运行结果如下:
在这里插入图片描述
3. 函数返回值类型为类类型对象
在这里插入图片描述
结论:为了提高程序效率,一般对象传参时,尽量使用引用类型,返回时根据实际场景,能用引用尽量使用引用。

5. 赋值运算符重载

5.1 运算符重载

C++为了增强代码的可读性引入了运算符重载运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。有了运算符重载以后,可以根据需求使得自定义类型的对象也可以像内置类型一样使用操作符。
函数名字为:关键字operator后面接需要重载的运算符符号
函数原型:返回值类型 operator操作符(参数列表)
关于运算符重载,有一下几点需要注意:

  1. 不能通过连接其他符号来创建新的操作符:比如operator@ 。只能重载现有的操作符,比如 operator+,operator+=等。
  2. 重载操作符必须有一个类类型参数。
  3. 用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义。
  4. 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this。
  5. .* :: sizeof ?: . 注意以上5个运算符不能重载。

举个例子:比如想比较两个日期是否相等。
之前我们想比较两个日期的大小需要写一个比较大小的函数,现在有了运算符重载,我们可以重载运算符 ’ ==’ 使得Date类的对象可以像自定义类型一样使用运算符’ ==’ 来判断两个日期是否相等。

#include <iostream>
using namespace std;
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}
//private:int _year;int _month;int _day;
};
//运算符 == 重载,重载成全局的
bool operator==(const Date& d1, const Date& d2)
{return d1._year == d2._year&& d1._month == d2._month&& d1._day == d2._day;
}
int main()
{Date d1(2024, 2, 6);Date d2(2024, 2, 7);cout << (d1 == d2) << endl;//这里d1 和 d2可以像内置类型一样使用操作符return 0;
}

代码运行结果:
在这里插入图片描述
上面将运算符重载成全局的有个很大的问题(上面能运行是因为将成员变量的访问限定符改成了public),就是运算符重载成全局的就需要成员变量是公有的,那么问题来了,封装性如何保证?
这里其实可以用我们后面学习的友元来解决,或者干脆重载成成员函数
在这里插入图片描述
因此正确的写法为:

class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}// 运算符 == 重载,重载成成员函数// bool operator==(Date* this, const Date& d)// 这里需要注意的是,左操作数是this,指向调用函数的对象bool operator==(const Date& d){return _year == d._year&& _month == d._month&& _day == d._day;}
private:int _year;int _month;int _day;
};
int main()
{Date d1(2024, 2, 6);Date d2(2024, 2, 7);cout << (d1 == d2) << endl;//这里d1 和 d2可以像内置类型一样使用操作符//上面那样写是为了提高代码的可读性,我们也可以像下面这样显示的调用cout << (d1.operator==(d2)) << endl;//bool operator==(Date* this, const Date& d2)return 0;
}

总结:运算符函数可以定义为类的成员函数或全局函数。如果运算符函数是类的成员函数,它将自动获得一个隐含的 this 指针,用于访问调用对象的成员;如果是全局函数,则需要在参数列表中显式地传递所有操作数。

5.2 赋值运算符重载

  1. 赋值运算符重载是指重载类中的赋值运算符(=),使得用户能够对自定义类型的对象进行赋值操作。通过赋值运算符重载,可以实现类对象之间的拷贝。
    语法格式为:
    参数类型:const T&,传递引用可以提高传参效率。
    返回值类型:T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值。
    检测是否自己给自己赋值
    返回*this :要复合连续赋值的含义。
//T是类型名
T& operator=(const T& 变量名) 
{// 执行赋值操作// 返回 *this
}
  1. 赋值运算符只能重载成类的成员函数不能重载成全局函数。
    在这里插入图片描述
    正确做法如下:

在这里插入图片描述在这里插入图片描述
3. 用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝。注意:内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符重载完成赋值。
在这里插入图片描述
既然编译器生成的默认赋值运算符重载函数已经可以完成字节序的值拷贝了,还需要自己实现吗?当然像日期类这样的类是没必要的。像前面我们介绍拷贝构造函数那里的Stack类一样,我们发现程序就会崩溃掉。
原因如下:
在这里插入图片描述
总结:如果类中未涉及到资源管理(动态申请空间),赋值运算符是否实现都可以;一旦涉及到资源管理赋值运算符则必须要实现。

6. const成员函数

使用const修饰的“成员函数”称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。并且在函数声明和定义中都需要加上 const 关键字。
语法如下:

返回类型 函数名() const 
{// 函数体
}

在这里插入图片描述
经过上面的介绍,我们来思考如下几个问题:

  1. const对象可以调用非const成员函数吗?
  2. 非const对象可以调用const成员函数吗?
  3. const成员函数内可以调用其它的非const成员函数吗?
  4. 非const成员函数内可以调用其它的const成员函数吗?

在这里插入图片描述

7. 取地址及const取地址操作符重载

这两个默认成员函数一般不用重新定义 ,编译器默认会生成。

class Date
{
public:Date* operator&(){return this;}const Date* operator&() const{return this;}
private:int _year;int _month;int _day;
};

这两个运算符一般不需要重载,使用编译器生成的默认取地址的重载即可,只有特殊情况,才需
要重载,比如想让别人获取到指定的内容

至此,本片文章就结束了,若本篇内容对您有所帮助,请三连点赞,关注,收藏支持下。

创作不易,白嫖不好,各位的支持和认可,就是我创作的最大动力,我们下篇文章见!

如果本篇博客有任何错误,请批评指教,不胜感激 !!!
在这里插入图片描述

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

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

相关文章

2024-2-6-复习作业

1> 要求&#xff1a; 源代码&#xff1a; #include <stdio.h> #include <stdlib.h> void output(int arr[],int len) {for(int i0;i<len;i){printf("%d ",arr[i]);}puts(""); } void bubble_sort(int arr[],int len) {for(int i1;i<…

C++类和对象(6)

目录 1. 再谈构造函数 1.1 构造函数体赋值 1.2 初始化列表 1.3 explicit关键字 2. static成员 2.1 概念 2.2 特性 【问题】 1. 再谈构造函数 1.1 构造函数体赋值 在创建对象时&#xff0c;编译器通过调用构造函数&#xff0c;给对象中各个成员变量一个合适的初始值。 c…

python的数据类型

&#x1f388;srting&#xff08;字符串&#xff09;&#xff1a; 操作符&#xff1a; &#xff1a;字符串连接 aabc befg print(ab) #输出 abcdefg * : 重复输出字符串 aabc print(a*3) #输出 abcabcabc [ : ]:截取字符串中的一部分&#xff0c;遵循左闭右开的原则&am…

迭代器和生成器

迭代器和生成器 一、迭代器① iter()② next()③ 自定义迭代器 二、生成器① 创建生成器1、斐波那契数列2、yield 创建 ② 使用send() 一、迭代器 迭代器是一个可以记住遍历的位置的对象&#xff0c;迭代器从第一个元素开始访问&#xff0c;直到所有元素访问结束 ① iter() …

Vue3快速上手(二)VSCode官方推荐插件安装及配置

一、VSCode官方插件安装&#xff0c;如下图2款插件 在用vite创建的程序里&#xff0c;提示提安装推荐的插件了&#xff0c;如下图&#xff1a; 二、配置 在设置-扩展里找到Volar插件&#xff0c;将Dot Value勾选上。这样在ref()修改变量时&#xff0c;会自动填充.value,无需…

电力负荷预测 | Matlab实现基于LSTM长短期记忆神经网络的电力负荷预测模型(结合时间序列)

文章目录 效果一览文章概述源码设计参考资料效果一览 文章概述 电力负荷预测 | Matlab实现基于LSTM长短期记忆神经网络的电力负荷预测模型(结合时间序列) 所谓预测,就是指通过对事物进行分析及研究,并运用合理的方法探索事物的发展变化规律,对其未来发展做出预先估计和判断…

李宏毅LLM——大模型+大资料的神奇力量

文章目录 大模型的重要性顿悟时刻 大资料的重要性数据预处理不一样的做法&#xff1a;KNN LM 对应视频P12-P14 大模型的重要性 模型参数和数据集越大&#xff0c;文字接龙的错误率越低 顿悟时刻 当模型超过10B-20B时&#xff0c;会突然顿悟 启示&#xff1a;不能只看最终结…

vue3:24—组件通信方式

目录 1、props 2、自定义事件 &#xff08;emit&#xff09; 3、mitt&#xff08;任意组件的通讯&#xff09; 4、v-model【封装ui组件库用的多&#xff0c;平时用的少。和vue2有点不同】 5、$attrs 6、$refs和$parent 7、provide和inject 8、pinia&#xff08;即vue2中…

HTML 样式学习手记

HTML 样式学习手记 在探索网页设计的世界时&#xff0c;我发现HTML元素的样式调整真的是个很酷的环节。通过简单的属性设置&#xff0c;就能让文字换上五彩斑斓的颜色、变换各异的字体和大小。特别是那个style属性&#xff0c;感觉就像是一扇通往CSS魔法世界的大门。 代码小试…

c语言实现io多路复用(select),进程,线程并发服务器

io多路复用&#xff08;select&#xff09;代码 #include<myhead.h> #include <sys/select.h> #define PORT 8888 #define IP "192.168.250.100" int main(int argc, char const *argv[]) { //创建套接字int sfd socket(AF_INET, SOCK_STREAM, 0…

常见的 MIME(媒体)类型速查

一、简介 MIME(Multipurpose Internet Mail Extensions)多用途互联网邮件扩展类型&#xff0c;是设定某种扩展名的文件用一种应用程序来打开的方式类型&#xff0c;当该扩展名文件被访问的时候&#xff0c;浏览器会自动使用指定应用程序来打开。多用于指定一些客户端自定义的文…

P1808 单词分类

P1808 单词分类 题目描述 Oliver 为了学好英语决定苦背单词&#xff0c;但很快他发现要直接记住杂乱无章的单词非常困难&#xff0c;他决定对单词进行分类。 两个单词可以分为一类当且仅当组成这两个单词的各个字母的数量均相等。 例如 AABAC&#xff0c;它和 CBAAA 就可以…

时序预测 | MATLAB实现基于CNN-BiLSTM-AdaBoost卷积双向长短期记忆网络结合AdaBoost时间序列预测

时序预测 | MATLAB实现基于CNN-BiLSTM-AdaBoost卷积双向长短期记忆网络结合AdaBoost时间序列预测 目录 时序预测 | MATLAB实现基于CNN-BiLSTM-AdaBoost卷积双向长短期记忆网络结合AdaBoost时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 1.Matlab实现…

SpringBoot3整合Mybatis-Plus,自定义动态数据源starter

文章目录 前言正文一、项目总览二、核心代码展示2.1 自定义AbstractRoutingDataSource2.2 动态数据源DynamicDataSource2.3 动态数据源自动配置2.4 动态数据源上下文DynamicDataSourceContextHolder2.5 动态数据源修改注解定义2.6 修改切面DynamicDataSourceAspect2.7 动态数据…

多维时序 | MATLAB实现基于CNN-LSSVM卷积神经网络-最小二乘支持向量机多变量时间序列预测

多维时序 | MATLAB实现基于CNN-LSSVM卷积神经网络-最小二乘支持向量机多变量时间序列预测 目录 多维时序 | MATLAB实现基于CNN-LSSVM卷积神经网络-最小二乘支持向量机多变量时间序列预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.MATLAB实现基于CNN-LSSVM卷积神经…

vue+vite项目,动态导入静态资源的几种方式

博主的桌面工具软件已经正式开发&#xff0c;获取方式&#xff1a; 可以关注我的小程序【中二少年工具箱】获取。&#xff08;若小程序更新有延迟&#xff0c;可先收藏小程序&#xff09; 通过下载链接 百度网盘: 链接&#xff1a;https://pan.baidu.com/s/15zDnSoEzJGSZLjpD…

跟着pink老师前端入门教程-day20

二、移动WEB开发之flex布局 1、flex 布局体验 1.1 传统布局与flex布局 传统布局&#xff1a;兼容性好、布局繁琐、局限性、不能再移动端很好的布局 flex弹性布局&#xff1a;操作方便&#xff0c;布局极为简单&#xff0c;移动端应用很广泛&#xff1b;PC 端浏览器支持情况…

SpringbootV2.6整合Knife4j 3.0.3 问题记录

参考 https://juejin.cn/post/7249173717749940284 近期由于升级到springboot2.6X&#xff0c;所以服务端很多组件都需要重新导入以及解决依赖问题。 下面就是一个很经典的问题了&#xff0c; springboot2.6与knife4j的整合。 版本对应 springboot2.6与knife4j 3.0.3 坑 …

c++二叉树寒假特训题目(1)

大家好&#xff0c;我是周曦&#xff0c;今天给大家推荐一些二叉树题目。 题目 二叉树存储 这道题是道水题&#xff0c;找找规律ok&#xff0c;本人代码10行。 淘汰赛 这道题推荐使用桶数组 做比较合适&#xff08;就是有点绕&#xff09;。 二叉树深度 这题是一道深搜题&a…

eclipse使用google的Java代码格式

插件下载地址 1.下载eclipse的插件 2.下载的jar包放到eclipse安装目录的dropins文件夹 D:\install_package\STS\sts-4.10.0.RELEASE\dropins&#xff13;.重启后设置 eclipse - windows - preference - java - code style - formatter -