c++初阶——类和对象(中)

大家好,我是小锋,我们今天继续来学习类和对象。

类的6个默认成员函数

我们想一想如果一个类什么都没有那它就是一个空类,但是空类真的什么都没有吗?

其实并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。

默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。

我们下面来分别学习

构造函数

概念

我们先来写一个日期类

class date {
public:
	void  init(int yare = 1, int month = 1, int day = 1) {
		_yare = yare;
		_month = month;
		_day = day;
	}
	void Printf() {
		cout << _yare << "/" << _month << "/" << _day << "/" << endl;
	}
private:
	int _yare;
	int _month;
	int _day;
};
int main() {
	date add;
	add.init(2024, 4, 20);
	add.Printf();

	date arr;
	arr.init();
	arr.Printf();


	return 0;
}

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

这里就可以用到我们的构造函数了。

构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,以保证 每个数据成员都有 一个合适的初始值,并且在对象整个生命周期内只调用一次。

构造函数的特性

构造函数是特殊的成员函数,需要注意的是,构造函数虽然名称叫构造,但是构造函数的主要任 务并不是开空间创建对象,而是初始化对象。

其特征如下:

1. 函数名与类名相同。

2. 无返回值。

3. 对象实例化时编译器自动调用对应的构造函数。

4. 构造函数可以重载。

# include<iostream>
using namespace std;


class date {
public:
	date(int yare = 0, int month = 0, int day = 0) {
		_yare = yare;
		_month = month;
		_day = day;
	}

	void Printf() {
		cout << _yare << "/" << _month << "/" << _day << "/" << endl;
	}
private:
	int _yare;
	int _month;
	int _day;
};
int main() {
	//调用无参构造函数
	date add;
	add.Printf();
	//调用带参数的构造函数
	date arr(2024, 4, 20);
	arr.Printf();
	return 0;
}

我们还要注意一个问题

相当于我们没有定义app的类,而是做了一个函数声明。

5. 如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。

6. 关于编译器生成的默认成员函数,很多童鞋会有疑惑:不实现构造函数的情况下,编译器会生成默认的构造函数。但是看起来默认构造函数又没什么用?对象调用了编译器生成的默认构造函数,但是对象的_year/_month/_day,依旧是随机值。也就说在这里编译器生成的默认构造函数并没有什么用??

解答:C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char...,自定义类型就是我们使用class/struct/union等自己定义的类型,看看下面的程序,就会发现编译器生成默认的构造函数会对自定类型成员_t调用的它的默认成员函数。

编译器生成的默认构造函数对内置类型不处理,只处理自定义类型

class Time {
public:
	Time(int hour = 0, int minute = 0) {
		cout << "time(int hour = 0, int minute = 0)" << endl;
		_hour = hour;
		_minute = minute;
	}
private:
	int _hour;
	int _minute;
};
class date {
public:
	date(int yare = 0, int month = 0, int day = 0) {
		cout << "date(int yare = 0, int month = 0, int day = 0)" << endl;
		_yare = yare;
		_month = month;
		_day = day;
	}
	void Printf() {
		cout << _yare << "/" << _month << "/" << _day << "/" << endl;
	}
private:
	int _yare;
	int _month;
	int _day;
	Time _a;
};
int main() {
	date add;
	return 0;
}

从图中我们可以看到对于自定义类型Time编译器自动生成了构造函数。

注意:C++11 中针对内置类型成员不初始化的缺陷,又打了补丁,即:内置类型成员变量在 类中声明时可以给默认值。

意思是我们可以这样初始化

这样编译器生成的默认构造函数就会对内置类型进行处理了

绝大部分的类要自己写构造函数,只有自定义类型的对象不用写构造函数,编译器会自己调用默认的构造函数。

7. 无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。 注意:无参构造函数、全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数。

析构函数

概念

通过前面构造函数的学习,我们知道一个对象是怎么来的,那一个对象又是怎么没呢的?

析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由 编译器完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作。

析构函数的特性

析构函数是特殊的成员函数,其特征如下:

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

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

3. 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载(同样的编译器自动生成的析构函数只对自定义类型处理

4. 对象生命周期结束时,C++编译系统系统自动调用析构函数。

我们简单来写个栈

# include<assert.h>
class stack {
public:
	stack(int n=4) {
		int* cur = (int*)malloc(sizeof(int) * n);
		if (cur == nullptr) {
			perror("空间开辟失败");
			return;
		}
		_a = cur;
		_size = 0;
		_capacity = n;
	}
	void push(int mon) {
		if (_size == _capacity) {
			int* cur = (int*)realloc(_a, sizeof(int) * _capacity * 2);
			if (cur == nullptr) {
				perror("空间开辟失败");
				return;
			}
			_a = cur;
			_capacity *= 2;
		}
		_a[_size] = mon;
		_size++;
	}
	void Printf() {
		for (int j = 0; j < _size; j++) {
			cout << _a[j] << endl;
		}
	}
	bool Empty() {
		return _size == 0;
	}
	void pop() {
		assert(!Empty());
		 _size--;
	}
	~stack() {
		free(_a);
		_a = nullptr;
		_size = 0;
		_capacity = 0;
	}

private:
	int* _a;
	int _size;
	int _capacity;
};


int main() {
	stack head;
	head.push(1);
	head.push(2);
	head.push(3);
	head.push(4);
	head.push(5);
	head.pop();
	head.Printf();
	return 0;
}

大家看是不是程序结束是编译器自动调用析构函数

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

编译器自动生成的默认析构函数对自定义类型对象的处理

6. 如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数;有资源申请时,一定要写,否则会造成资源泄漏。

拷贝构造函数

概念

拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存 在的类类型对象创建新对象时由编译器自动调用。

同学们我们想想拷贝构造函数的意义何在,我们在学习c语言时也没用拷贝构造啊?

我们来探讨一下大家跟紧我的思路

# include<iostream>
using namespace std;

class date {
public:

	date(int year=1, int month=1, int day=1) {
		_year = year;
		_month = month;
		_day = day;
	}
	void Printf() {
		cout << _year << "年" << _month << "月" << _day << "日" << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};

int main() {
	date d1;
	d1.Printf();
	date d2(2024, 4, 23);
	d2.Printf();
	return 0;
}

我们还是简单的来实现一个日期类

大家看我们这里定义了一个函数fund1在传参时是不是发生了拷贝这里是简单的值拷贝。

我们发现这里都是内置类型的拷贝,那么自定义类型可以这样拷贝吗?

我们来简单实现一个栈来演示

class stack {
public:
	stack(int n=4) {
		datapety* cur = (datapety*)malloc(sizeof(datapety) * n);
		if (cur == nullptr) {
			perror("空间开辟失败");
			return;
		}
		_add = cur;
		_size = 0;
		_capacity = n;
	}
	void push(int n) {
		if (_size == _capacity) {
			datapety* cur = (datapety*)realloc(_add,sizeof(datapety) * _capacity * 2);
			if (cur == nullptr) {
				perror("扩容失败");
					return;
			}
			_add = cur;
			_capacity *= 2;
		}
		_add[_size] = n;
		_size++;
	}
	bool Empty() {
		return _size == 0;
	}


	void pop() {
		assert(!Empty());
		_size--;
	}
	void Printf() {
		for (int i = 0; i < _size; i++) {
			cout << _add[i];
		}
		cout << endl;
	}

	~stack() {
		free(_add);
		_add = nullptr;
		_size = 0;
		_capacity = 0;
	}

private:
	datapety* _add;
	int _size;
	int _capacity;

};

当我们这样拷贝时程序崩溃了

那这是为什么呢?

这里崩溃的原因是空间连续销毁了两次,我们来分析一下我们的代码

大家看我们在调用fund2时是采用值拷贝,将实参的值拷贝给形参,_add指针存放的是一个指向动态开辟空间的地址,在值拷贝时,将值拷贝,当fund2函数结束时要销毁栈帧,就会将_add指针指向的空间一起销毁,然后当我们的程序走完后又会自动调用析构函数,又会将这块空间再次free一次,所以这里程序崩溃了。

那我们这里应该怎么办呢?

这里就要用到我们学过的引用了

大家看这样程序是不是就可以这次运行并结束了?

但是我们引用又会造成一个问题

大家看我们这里对a进行push,a1也会改变,但是我们不行让它改变啊,我们应该这么做,

那就要用到我们的拷贝构造函数进行深拷贝了。

stack(stack& a) {
		_add = (datapety*)malloc(sizeof(datapety) * a._capacity);
		if (_add == nullptr) {
			perror("空间开辟失败");
			return;
		}
		for (int i = 0; i < a._size; i++) {
			_add[i] = a._add[i];
		}
		_size = a._size;
		_capacity = a._capacity;
	}

这样我们的a1和a就是两个互不关联但存储的数据相同的两个类了

特征

拷贝构造函数也是特殊的成员函数,其特征如下:

1. 拷贝构造函数是构造函数的一个重载形式。

2. 拷贝构造函数的参数只有一个且必须是类类型对象的引用(这里可以加一个const防止我们在写拷贝构造的函数时不小心修改了原对象),使用传值方式编译器直接报错, 因为会引发无穷递归调用。(这里编译器是强制报错的)

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

注意:在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,而自定义类型是调用其拷贝构造函数完成拷贝的。

4. 编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了

注意:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请 时,则拷贝构造函数是一定要写的,否则就是浅拷贝。

5. 拷贝构造函数典型调用场景:

使用已存在对象创建新对象

函数参数类型为类类型对象

函数返回值类型为类类型对象

赋值运算符重载

引出运算符重载

大家可以看到内置类型可以直接使用运算符,进行运算比较,那自定义类型可以直接比较吗?

很显然是不行的

我们应该写一个比较函数来进行比较

我们拿日期类来举例

我们写好函数后发现x1,x2无法访问到类的成员变量,这是因为我们在定义成员变量时是限定私有的,我们只要改成公有的或者直接把比较函数改成类的成员函数就解决了。

这里,我们还有一个问题,我们的函数名时拼音,可读性很差,改成其他的名称也不是很好,所以为了规范,我们c++祖师爷引入了运算符重载

运算符重载

C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其 返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。

函数名字为:关键字operator后面接需要重载的运算符符号。

函数原型:返回值类型 operator操作符(参数列表)

注意:

1,不能通过连接其他符号来创建新的操作符:比如operator@

2,重载操作符必须有一个类类型参数

3,用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义

4,作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this

5,.*   ::   sizeof    ? :    . 注意以上5个运算符不能重载。这个经常在笔试选择题中出现。

6,不能改变操作符的操作数个数(就是有几个操作数重载时就有几个参数)

我们对上面代码更改后就变成了下面的样子

bool operator< (const date& d) {
		if (_year < d._year) {
			return true;
		}
		else if (_year == d._year && _month < d._month) {
			return true;
		}
		else if (_year = d._year && _month == d._month && _day < d._day) {
			return true;
		}
		else {
			return false;
		}
	}

这里我们的参数用了const,是因为比较大小是不用改变对象的值的,用引用是在传参时不用进行拷贝提高效率。

# include<iostream>
using namespace std;

class date {
public:
	date(int year = 1, int month = 1, int day = 1) {
		_year = year;
		_month = month;
		_day = day;
	}
	void Printf() {
		cout << _year << "年" << _month << "月" << _day << "日" << endl;
	}

	bool operator< (const date& d) {
		if (_year < d._year) {
			return true;
		}
		else if (_year == d._year && _month < d._month) {
			return true;
		}
		else if (_year = d._year && _month == d._month && _day < d._day) {
			return true;
		}
		else {
			return false;
		}
	}
private:
	int _year;
	int _month;
	int _day;
};
int main() {
	date d1(2018, 7, 9);
	date d2(2024, 4, 24);
	cout << (d1 < d2) << endl;

}

这里我们是不是就比较成功了?

接下来我们来看看赋值运算符重载是怎么样的

我们看这里我们定义了两个日期类,我们现在想把d2赋值给d1,应该怎么赋值,首先这里是自定义类型的赋值,看到不能直接用=进行赋值,所以我们这里就要用到我们刚刚学的运算符重载,对赋值运算符进行重载

date& operator=(const date & d) {
			if (this != &d) {
				_year = d._year;
				_month = d._month;
				_day = d._day;
			}
		}

大家对比是不是发现了这个赋值运算符重载函数与拷贝构造函数简直就是一模一样,实现的思路也是一样的,那它们两个函数的区别是什么呢?

首先赋值是两个已经存在的对象进行拷贝,而拷贝构造是一个已经存在的对象对另一个要创建的对象进行初始化。

赋值运算符主要有四点:

1. 参数类型

2. 返回值

3. 检测是否自己给自己赋值

4. 返回*this

5. 一个类如果没有显式定义赋值运算符重载,编译器也会生成一个默认的赋值函数,完成对象按字节序的值拷贝。

如果编译器生成的默认赋值重载函数已经可以完成字节序的值拷贝了,我们还需要自己实现吗?

答案当然是一定的,我们在拷贝构造函数时已经讲过了,如果只进行值拷贝,这是一种浅拷贝对于那些动态开辟的空间的内容是拷贝不了的,在退出函数时就会多次调用析构函数。

我们来完成一个完整的日期类,

日期类的实现

检验一下大家的学习成果

还是老规矩,我们还是用三个文件来分装不同的内容

date.h

#pragma once

# include<iostream>

using namespace std;

class date{
public:
	//构造函数
	date(int year = 1, int month = 1, int day = 1);
	//打印日期
	void Printf();
	//重载<
	bool operator<(const date& d);
	//重载==
	bool operator==(const date& d);
	//重载<=
	bool operator<=(const date& d);
	//重载>
	bool operator>(const date& d);
	//重载>=
	bool operator>=(const date& d);
	//重载!=
	bool operator!=(const date& d);

	//判断这个月有多少天
	int GetMonthDay(const int year, const int month);
	//重载+=
	date& operator+=(int day);
	//重载+
	date operator+(int day);
	//重载-=
	date& operator-=(int day);
	//重载-
	date operator-(int day);
	//重载前置++
	date& operator++();
	//重载后置++
	date& operator++(int);
	//重载前置--
	date& operator--();
	//重载后置--
	date& operator--(int);


private:
	int _year;
	int _month;
	int _day;
};

date.c

# include"date.h"


//构造函数
date::date(int year, int month, int day) {
	_year = year;
	_month = month;
	_day = day;
	//检查日期是否合法
	if ((month < 1 || month>12) || (day<1 || day>GetMonthDay(year, month))) {
		cout << "非法日期" << endl;
	}
}

//打印日期
void date::Printf() {

	cout << _year << "年" << _month << "月" << _day << "日" << endl;

}

//重载<
bool date::operator<(const date& d) {
	if (_year < d._year) {
		return true;
	}
	else if (_year == d._year && _month < d._month) {
		return true;
	}
	else if (_year == d._year && _month == d._month && _day < d._day) {
		return true;
	}
	else {
		return false;
	}
}

//重载==
bool date::operator==(const date& d) {
	if (_year == d._year && _month == d._month && _day == d._day) {
		return true;
	}
	else {
		return false;
	}
}

//重载<=
bool date::operator<=(const date& d) {
	return (*this < d) || (*this == d);

}

//重载>
bool date::operator>(const date& d) {
	return !(*this <= d);
}

//重载>=
bool date::operator>=(const date& d) {
	return !(*this < d);

}

//重载!=
bool date::operator!=(const date& d) {
	return !(*this == d);
}


//判断这个月有多少天
int date::GetMonthDay(const int year, const int month) {
	int add[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31, };
	int n = add[month];
	if (month == 2 && ((year % 4 == 0) && (year % 100 != 0)) || (year % 400 == 0)) {
		n = 29;
	}
	return n;
}

//重载+=
date& date::operator+=(int day) {
	if (day < 0) {
		return *this -= (-day);
	}
	_day += day;
	while (_day > GetMonthDay(_year, _month)) {
		_day -= GetMonthDay(_year, _month);
		_month++;
		if (_month == 13) {
			_year++;
			_month = 1;
		}
	}
	return *this;
}

//重载+
date date::operator+(int day) {
	date cmp (*this);
	cmp += day;
	return cmp;
}

//重载-=
date& date::operator-=(int day) {
	if (day < 0) {
		return *this += (-day);
	}
	_day -= day;
	while (_day <= 0) {
		_month--;
		if (_month == 0) {
			_year--;
			_month = 12;
		}
		_day += GetMonthDay(_year, _month);
	}
	return *this;
}

//重载-
date date::operator-(int day) {
	date tmp(*this);
	tmp -= day;
	return tmp;
}

//重载前置++
date& date::operator++() {
	*this += 1;
	return *this;
}
//重载后置++
date& date::operator++(int) {
	date cmp(*this);
	*this += 1;
	return cmp;
}
//重载前置--
date& date::operator--() {
	*this -= 1;
	return *this;
}
//重载后置--
date& date::operator--(int) {
	date tmp(*this);
	*this -= 1;
	return  tmp;
}

test.c

# include"date.h"


int main() {
	date d1(2023, 2, 23);
	date d2(2024, 4, 24);
	d1.Printf();
	d2.Printf();
	cout << (d1 < d2) << endl;
	cout << (d1 <= d2) << endl;
	cout << (d1 > d2) << endl;
	cout << (d1 >= d2) << endl;
	cout << (d1 == d2) << endl;
	cout << (d1 != d2) << endl;
	d1 += 10;
	d1.Printf();
	d1 -= 10;
	d1.Printf();
	d2 = d1 + 10;
	d2.Printf();
	d2 = d1 - 10;
	d2.Printf();
	++d2;
	d2.Printf();
	d1 = d2++;
	d1.Printf();
	d2.Printf();
	--d2;
	d2.Printf();
	d1 = d2--;
	d1.Printf();
	d2.Printf();


	return 0;
}

我们最后可以测试一下

这里主要讲一下前置++,后置++,以及前置--,后置--,在重载时默认是前置,后置则要在参数中加一个int参数,进行占位,跟前置构成函数重载进行区分,

日期-日期

我们要计算两个日期之间的天数就要用到这个函数

这里我们只要复用我们前面写的++函数就可以很简单搞定了

//日期相减
int date::operator - (const date& d) {
	date max = *this;
	date min = d;
	int fin = 1;
	if (min > max) {
		max = d;
		min = *this;
	}
	int n = 0;
	while (min != max) {
		min++;
		n++;
	}

	return n * fin;


}

我们测试一下

const成员

我们来看这段代码

这里我们用const修饰的d1对象无法调用类的成员函数,这是为什么呢?

我们知道类在调用成员函数时有一个this指针,它是指向调用函数的对象的,它的类型是date*的类型而我们调用函数的对象是从const date的类型,所以无法调用,那怎么才能调用呢?

肯定要将this指针的类型也用const 修饰,

接下来我们看看怎么修饰

const修饰类的成员函数

将const修饰的类成员函数称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。

这样修改后我们就可以调用类的成员函数了

总结

1. const对象不可以调用非const成员函数

2. 非const对象可以调用const成员函数

3. const成员函数内不可以调用其它的非const成员函数

4. 非const成员函数内可以调用其它的const成员函数

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

我们知道理论上c++的操作符只能对内置类型进行处理对自定义类型是不进行处理的,但我们来看这样一段代码

我们发现这里的&操作符对自定义类型的对象取地址了,这就是编译器默认的取地址重载,取出的是对象本来的地址

它的实现类似于如下(有两版本一个是const 一个是非const两个函数形成重载)

这个函数绝大部分情况使用编译器自己生成就已经够用了,真要我们自己写,大部分是不想别人取到我们的地址,一般给空

我们还可以给一些随机的值

 以上就是全部内容了,如果有错误或者不足的地方欢迎大家给予建议。 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++ | Leetcode C++题解之第43题字符串相乘

题目&#xff1a; 题解&#xff1a; class Solution { public:string multiply(string num1, string num2) {if (num1 "0" || num2 "0") {return "0";}int m num1.size(), n num2.size();auto ansArr vector<int>(m n);for (int i …

比亚迪海洋网再添实力爆款,海豹06DM-i、OCEAN-M、海狮07EV登陆北京车展

4月25日&#xff0c;比亚迪海洋网携海豹06DM-i、OCEAN-M、海狮07EV一齐亮相北京车展&#xff0c;引发关注热潮。其中&#xff0c;海洋网全新中型轿车海豹06DM-i价格区间12万-15万元&#xff0c;将于今年二季度上市&#xff1b;行业首款两厢后驱纯电钢炮OCEAN-M价格区间15万-20万…

重发布的原理及其应用

重发布的作用&#xff1a; 在一个网络中&#xff0c;若运行多种路由协议或者相同协议的不同进程&#xff1b;因为协议之间不能直接沟通计算&#xff0c;进程之间也是独立进行转发和运算的&#xff0c;所以&#xff0c;需要使用重发布来实现路由的共享。 条件 &#xff1a; 1&am…

RH850F1KM 搭建MCAL配置环境

1 搭建环境所需的安装包 安装以下软件安装包 1&#xff09;MCAL下载&#xff1a;AUTOSAR RH850/F1KM MCAL v42.11.00 Software 2&#xff09;Davinci工具 SIP包&#xff1a;DaVinci Configurator Pro 3&#xff09;下载网址&#xff1a;https://www.renesas.cn/cn/zh/product…

美团面试题-Nacos配置中心动态刷新原理

①&#xff1a;pull模式&#xff1a;主动拉去配置&#xff0c;通过固定的时间间隔。缺点&#xff1a;频繁请求&#xff0c;时效性不高&#xff0c;时间间隔不好设置。 ②&#xff1a;push模式&#xff1a;服务端检测到变化&#xff0c;主动将新配置推送给客户端&#xff0c;时效…

python的列表生成式

什么是列表生成式&#xff1f; 列表生成式&#xff0c;顾名思义&#xff0c;是生成列表的一个简单又直接的方法。它使用了一种紧凑的语法来构造列表&#xff0c;能够以一种更清晰、更简洁的方式来表达循环和过滤逻辑。 基础示例让我们先从一些简单的例子开始&#xff1a; 「生…

echarts 基础入门

前言&#xff1a;本文档主要讲解 echarts 在 vue3 中的用法&#xff0c;及其 echarts 的一些配置参数含义及用法。示例参考 echarts 示例 一&#xff1a;快速开始 1. 安装 echarts npm install echarts # or pnpm add echarts # or yarn add echarts 2. 使用 echarts <…

通过商品id采集京东商品详情数据(含价格、优惠券、详情、主图等字段)

item_get_app-获得JD商品详情原数据 公共参数 名称类型必须描述keyString是调用key&#xff08;注册账号获取测试key&#xff09;secretString是调用密钥api_nameString是API接口名称&#xff08;包括在请求地址中&#xff09;[item_search,item_get,item_search_shop等]cach…

【八股】Spring篇

why Spring? 1.使用它的IOC功能&#xff0c;在解耦上达到了配置级别。 2.使用它对数据库访问事务相关的封装。 3.各种其他组件与Spring的融合&#xff0c;在Spring中更加方便快捷的继承其他一些组件。 IoC和DI &#x1f449;IOC是Inversion of Control的缩写&#xff0c;“…

Day4 商品管理

Day4 商品管理 这里会总结构建项目过程中遇到的问题&#xff0c;以及一些个人思考&#xff01;&#xff01; 学习方法&#xff1a; 1 github源码 文档 官网 2 内容复现 &#xff0c;实际操作 项目源码同步更新到github 欢迎大家star~ 后期会更新并上传前端项目 编写品牌服务 …

【C++】string类的增删改查模拟实现(图例超详细解析!!!)

目录 一、前言 二、string类的模拟实现 ✨前情提要 ✨Member functions —— 成员函数 ⚡构造函数 ⚡拷贝构造函数 ⚡赋值运算符重载 ⚡析构函数 ✨Element access —— 元素访问 ⚡operator[ ] ⚡Iterator —— 迭代器 ✨Capacity —— 容量 ⚡size ⚡capacity ⚡clea…

2024年 Java 面试八股文(20w字)

目录 第一章-Java基础篇 1、你是怎样理解OOP面向对象 难度系数&#xff1a;⭐ 2、重载与重写区别 难度系数&#xff1a;⭐ 3、接口与抽象类的区别 难度系数&#xff1a;⭐ 4、深拷贝与浅拷贝的理解 难度系数&#xff1a;⭐ 5、sleep和wait区别 难度系数&a…

VSCode配置Eclipse快捷键

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、下载扩展二、使用扩展总结 前言 有时候我们可能会同时使用很多种IDE&#xff0c;每种IDE又有不同的Keymap&#xff0c;快捷键用起来很头疼。比如我&#x…

免费语音转文字:自建Whisper,贝锐花生壳3步远程访问

Whisper是OpenAI开发的自动语音识别系统&#xff08;语音转文字&#xff09;。 OpenAI称其英文语音辨识能力已达到人类水准&#xff0c;且支持其它98中语言的自动语音辨识&#xff0c;Whisper神经网络模型被训练来运行语音辨识与翻译任务。 此外&#xff0c;与其他需要联网运行…

Ubuntu 20.04.6下载、安装

一、下载 下载地址&#xff1a;https://cn.ubuntu.com/download 下载版本&#xff1a;ubuntu-20.04.6-desktop-amd64.iso 二、安装 参考博客&#xff1a; https://blog.csdn.net/lhl_blog/article/details/123406322 https://www.cnblogs.com/fieldtianye/p/17879840.html…

如何使用docker部署前端项目

账号:root 密码:*** 主机:ip地址 登录后: 初级名词: sudo 是Linux和类Unix操作系统中的一个命令,全称为“super user do”,用于允许普通用户以超级用户(root)的身份执行某些或全部命令 需要下载的软件: sudo yum install 软件名 sudo yum install lrzsz 上传软件 s…

婴儿专用洗衣机有必要吗?四大宝藏婴儿洗衣机测评对比

对于有了宝宝的家庭来说&#xff0c;洗衣成为了一项重要的家务事。大家都知道&#xff0c;宝宝的皮肤比较娇嫩&#xff0c;容易受到各种细菌、病毒的侵扰。所以&#xff0c;宝宝的衣物应该与大人的分开洗。婴儿洗衣机作为一种专门为婴幼儿家庭设计的洗衣机&#xff0c;其具有除…

Spring5深入浅出篇:Spring对象属性注入详解

Spring5深入浅出篇:Spring对象属性注入详解 很多粉丝私信我这个Spring5的课程在哪看,这边是在B站免费观看欢迎大家投币支持一下. https://www.bilibili.com/video/BV1hK411Y7zf 首先需要了解什么是注入?在创建对象的过程中,不仅仅是创建一个对象还需要为对象的属性赋值.这一系…

Win11和WinRAR取消折叠菜单恢复经典菜单

这里写目录标题 前言1. Win11恢复经典右键菜单1.1 修改前1.2 恢复成经典右键菜单1.3 修改后1.4 想恢复怎么办&#xff1f; 2. WinRAR取消折叠菜单恢复经典菜单2.1 修改前2.2 修改恢复为经典菜单2.3 修改后2.4 想恢复怎么办&#xff1f; 前言 最近换回了Windows电脑&#xff0c…

SpringBoot学习(四)NoSQL、接口文档、远程调用、消息服务、Web安全、可观测性、AOT

文章目录 NoSQLRedis整合场景整合自动配置原理定制化序列化机制redis客户端 接口文档OpenAPI3架构整合使用常用注解Docket配置 远程调用WebClient创建与配置获取响应定义请求体 HTTP interface导入依赖定义接口创建代理&测试 消息服务消息队列-场景异步解耦削峰缓冲 消息队…