【C++初阶】之类和对象(中)

【C++初阶】之类和对象(中)

  • ✍ 类的六个默认成员函数
  • ✍ 构造函数
    • 🏄 为什么需要构造函数
    • 🏄 默认构造函数
    • 🏄 为什么编译器能自动调用默认构造函数
    • 🏄 自己写的构造函数
    • 🏄 构造函数的特性
  • ✍ 拷贝构造函数
    • 🏄 编译器默认生成的拷贝构造函数
    • 🏄 自己写的拷贝构造函数
    • 🏄 拷贝构造函数调用的场景
  • ✍ 赋值运算符重载(也叫拷贝赋值函数)
    • 🏄 运算符重载的引入
      • 💘 前置++和后置++重载
      • 💘 运算符重载函数的调用
    • 🏄 赋值运算符重载
      • 💘 编译器默认生成的赋值运算符重载函数
      • 💘 自己显示写的赋值运算符重载函数
  • ✍ 析构函数
    • 🏄 编译器默认生成的析构函数
    • 🏄 显式写的析构函数
    • 🏄 析构函数的特性
    • 🏄 没有深拷贝,导致二次释放同一空间问题
      • 💘 问题的引入---拷贝构造函数
      • 💘 问题的解决---深拷贝
      • 💘 赋值运算符重载函数的浅拷贝问题
  • ✍ const成员函数
    • 🏄 const对象访问的规则
    • 🏄 非const对象访问的规则
  • ✍ 对普通对象的取地址运算符重载和对const对象取地址运算符重载
  • ✍ C++默认构造函数提供的机制
    • 🏄 C++默认构造函数是否提供的情况

📃博客主页: 小镇敲码人
💞热门专栏:C++初阶
🚀 欢迎关注:👍点赞 👂🏽留言 😍收藏
🌏 任尔江湖满血骨,我自踏雪寻梅香。 万千浮云遮碧月,独傲天下百坚强。 男儿应有龙腾志,盖世一意转洪荒。 莫使此生无痕度,终归人间一捧黄。🍎🍎🍎
❤️ 什么?你问我答案,少年你看,下一个十年又来了 💞 💞 💞

✍ 类的六个默认成员函数

当类为空是编译器也不是什么都不生成,而是会生成六大默认成员函数。

在这里插入图片描述
我们也可以自己显式把这六个默认成员写出来,这样编译器就会调用我们自己的,而不会调用默认生成的。

✍ 构造函数

🏄 为什么需要构造函数

我们学习C语言的时候,初始化栈操作需要自己写一个Init函数,但是这样就很麻烦,因为初始化栈之后需要我们显示的去调用Init函数,否则就有可能出现野指针的情况,因为如果是链式的栈,要把next指针初始化为空。

🏄 默认构造函数

我们构造函数就是为了解决这样的问题,在初始化类的时候,
你不需要显示的调用Init函数,编译器会自动的去调用,如果你不去显示的写,

编译器会生成一个默认的构造函数。我们来验证一下。
class Date
{
private:int year;int day;int month;
};
int main()
{Date x;return 0;
}

此时我们写了一个Date类,编译器会给调用它的默认构造函数吗?运行结果:

在这里插入图片描述

怎么回事呢?x对象的值没有被初始化呀,那是不是代表编译器没有调用默认构造函数呢?其实不然,C++把类型分为自定义类型和内置类型,默认构造函数要做的是,自定义类型去调用它自己的构造函数(如果有的话),内置类型去给一个随机值,那到底是不是这样呢?我们也可以来验证一下。

class year
{
public:year(){std::cout << "year()" << std::endl;}
};
class Date
{
private:year y;int day;int month;
};
int main()
{Date x;return 0;
}

注意:那个自定义类型的构造函数必须是public的,否则在它自己的类外面就访问不了。

在这里插入图片描述
运行结果:

在这里插入图片描述

默认构造函数默认成员函数是两个不同的概念,两者不能混淆,不用我们传参数,全缺省构造函数和无参数构造函数、默认生成的构造函数都称作为默认构造函数。

class Date
{
public:Date(){day = 0;month = 0;}Date(int day = 0,int month = 0){}
private:year y;int day;int month;
};
int main()
{Date x;return 0;
}

注意:上面那两个默认构造函数不能同时存在,因为都不需要传参数,会造成歧义,编译器不知道调用哪一个默认构造函数。

在这里插入图片描述

🏄 为什么编译器能自动调用默认构造函数

那为什么编译器能在实例化类对象的的时候自动调用它的构造函数呢?

可以认为这是编译器做了特殊的处理,它帮助我们调用了这个函数。我们转到反汇编,可以发现编译器帮助我们调用了。

在这里插入图片描述

🏄 自己写的构造函数

我们也可以自己显示的写构造函数,那样编译器就不会去调用默认生成的构造函数了。


class year
{
public:year(){std::cout << "year()" << std::endl;}
};
class Date
{
public:Date(){day = 0;month = 0;}
private:year y;int day;int month;
};
int main()
{Date x;return 0;
}

运行结果:

在这里插入图片描述

可以看到,编译器在我们自己写的构造函数进去前,仍然会先去调用自定义类型的构造函数。

🏄 构造函数的特性

1、一次实例化对象只会调用一次,不支持显示调用。

在这里插入图片描述

2、构造函数会在实例化对象的时候自动调用,只用于初始化对象的一些成员变量,是初始化对象,而不是给对象开空间。

3、函数名和类名相同,无返回值。

4、支持重载。

✍ 拷贝构造函数

拷贝构造函数是构造函数的一种,主要作用是实现用一个已经存在的类对象,去初始化创建另外一个类对象。

🏄 编译器默认生成的拷贝构造函数

和前面普通的构造函数一样,如果我们不写编译器就会默认生成。

class year
{
public:year(){std::cout << "year()" << std::endl;}
};
class Date
{
public:Date(){day++;month++;}void f(){}
private:year y;int day = 0;int month = 0;
};
int main()
{Date x;Date y(x);return 0;
}

运行结果:

在这里插入图片描述

那我们为什么要写呢?这样岂不是浪费时间多次一举吗,编译器都帮助我们写好了,我们有时候的确是不需要写的,比如在没有向堆申请空间的时候,这时不涉及资源的清理,浅拷贝不会出问题,但是一旦我们向堆上申请空间后,不自己写深拷贝的拷贝构造函数,就会造成二次释放相同空间的问题。

🏄 自己写的拷贝构造函数

Date(const Date& x)
{day = x.day;month = x.month;y = x.y;
}

这里加上const是因为我们只是用x去初始化,但不希望改变它的值,至于这里为什么要使用引用,而且必须使用引用否则就会引发无穷递归:

在这里插入图片描述

这是因为我们在传参的时候,实参和形参的关系是,形参是实参的拷贝(当两者类型一样时),这不就相当于使用实参去初始化形参吗(也就是一个类去初始化另外一个类),也要调用拷贝构造函数,下一次又是一样的情况,所以会造成无穷递归,但是加了引用,你这个形参就是我实参的别名,不用再去调用拷贝构造,也就不会出现这种问题。

🏄 拷贝构造函数调用的场景

刚刚我们其实已经说了两个场景了。

1、用一个创建好的类初始化另外一个没有初始化的类
2、函数传参(参数为自定义类型)
3、函数返回值(参数为自定义类型),2和3都不能带引用,否则就不会调用拷贝构造函数。
4、赋值运算符重载时,被赋值的类还没有创建。

✍ 赋值运算符重载(也叫拷贝赋值函数)

我们的内置类型可以支持,一个变量赋值给另外一个对象,比如:a = b(都是int类型),那我们类(自定义类型)支持吗,答案是肯定的,使用运算符重载就可以解决这个问题。

🏄 运算符重载的引入

在C++中,增加了运算符的重载,这是因为有时候自定义类型也需要做一些类似操作符的操作,引入运算符重载,极大的提升了代码的可读性,它的规则如下:

  1. 函数名为operator后面接需要重载的运算符,注意:不能重载一些莫名奇妙的符号像@。
  2. 函数原型:返回值类型 operator操作符(参数列表)

注意:运算符重载时必须要有一个自定义类型的参数,因为运算符重载就是为类而生的,如果你没有类参数,那就没有意义了。

在这里插入图片描述
编译器为了防止你乱搞,会报错的。上面是全局的运算符重载函数。

有时候有的运算符需要两个参数,但是我们在类里面设计的时候只有一个参数,实际上是有两个参数的,第一个参数传的是this指针,编译器给隐藏了

💘 前置++和后置++重载

我们来介绍一下两个特殊的运算符重载,前置++和后置++重载,这两个操作符名字都一样,该如何区分呢?

这里没有办法了,C++对其做了特殊的处理,即给后置++多传一个参数来去区分,并且++操作符重载,最多额外传一个int参数作区分,也是为了防止用户乱搞。

我们来实现一下Date类的前置++和后置++:

// 前置++运算符  
// 该运算符将对象的年份、月份和日期都递增1,并返回递增后的对象的引用  
Date& operator++() // 前置++  
{  year++;       // 递增年份  day++;        // 递增日期  month++;      // 递增月份  return *this; // 返回当前对象的引用  
}  Date operator++(int) // 后置++  
{  Date tmp(*this); // 创建当前对象的副本  ++(*this);       // 递增当前对象(使用前置++)  return tmp;      // 返回递增前的对象的副本  
}

这里实际上我们在++日期的时候要考虑月份和年份的变化,这里我们主要是学习语法就不考虑了。
注意后置++的返回值不能带引用。因为我们返回的是副本,但是副本是临时对象(出了作用域销毁了),所以我们需要返回一个副本的拷贝,而不是副本本身。

  • 注意这里即使我们运算符重载函数写成全局的,也能像内置类型那样调用:
using namespace std;
class Date
{
public:Date(int year, int month = 2, int day = 1) ://普通的构造函数year_(year),month_(month),day_(day){cout << "Date(int year, int month = 2, int day = 1)" << endl;}Date(const Date& x) ://拷贝构造函数year_(x.year_),month_(x.month_),day_(x.day_){cout << "Date(const Date& x)" << endl;}Date& operator=(const Date& x)//拷贝赋值函数{year_ = x.year_;month_ = x.month_;day_ = x.day_;cout << "operator=(const Date& x)" << endl;return *this;}~Date()//析构函数{cout << "~Date" << endl;}
public:int year_;int month_;int day_;
};// 前置++运算符  
// 该运算符将对象的年份、月份和日期都递增1,并返回递增后的对象的引用  
Date& operator++(Date& x) // 前置++  
{x.year_++;       // 递增年份  x.day_++;        // 递增日期  x.month_++;      // 递增月份  return x; // 返回当前对象的引用  
}Date operator++(Date&x,int) // 后置++  
{Date tmp(x); // 创建当前对象的副本  ++x;       // 递增当前对象(使用前置++)  return tmp;      // 返回递增前的对象的副本  
}int main()
{Date x(2022);x++;++x;
}

运行结果:

在这里插入图片描述
代码正常运行。

如果我们给++运算符重载函数增加其它类型的参数,编译器就会报错:

在这里插入图片描述

💘 运算符重载函数的调用

内置类型可以直接a = b,或者a++,那我们的自定义类型是否可以这样了,为了可读性和方便,我们的C++支持这样来调用运算符重载函数,我们拿刚刚的前置++、和后置++函数来演示。

int main()
{Date x;x++;//-->operator++(&x,1);++x;//-->operator++(&x);return 0;
}

我们转到反汇编可以发现,确实是调用了对应的函数。

在这里插入图片描述
也可以显示调用,注意这里编译器已经帮助我们传了this指针过去,所以这里我们显示调用的是后置++:

在这里插入图片描述

🏄 赋值运算符重载

回归正题,我们继续来看我们的赋值运算符重载函数。

💘 编译器默认生成的赋值运算符重载函数

当我们不去显示的写赋值运算符重载函数,编译器会默认生成一个。

在这里插入图片描述

但是当我们这样去写,被赋值的y还没有被创建这个时候编译器就会去调用拷贝构造函数,无论你有没有自己显式的写

在这里插入图片描述

💘 自己显示写的赋值运算符重载函数

下面我们来自己显示的写一下,还是会有深拷贝的问题,当我们类的成员变量有在堆上申请空间时,直接赋值会引发二次析构的问题。

// 赋值运算符重载函数  
// 将参数x的值赋给当前对象,并返回当前对象的引用  
Date& operator=(const Date& x)  
{  if (this != &x) // 检查自赋值  {  day = x.day;  month = x.month;  year = x.year;  }  return *this;  
}

现代写法:

这种写法在拷贝构造函数处理好深拷贝问题后,可以很好的实现深拷贝,因为我们这种写法本质是对拷贝构造函数的一个复用。

Date& operator=(const Date& x)  
{  // 检查自赋值,避免不必要的操作  if (this != &x)  {  // 创建一个临时Date对象tmp,并使用参数x来初始化它  Date tmp(x);  // 使用std::swap来交换tmp对象的day成员和当前对象的day成员  std::swap(tmp.day, this->day);  // 使用std::swap来交换tmp对象的year成员和当前对象的year成员  std::swap(tmp.year, this->year);  // 使用std::swap来交换tmp对象的month成员和当前对象的month成员  std::swap(tmp.month, this->month);  // 通过上述交换,实际上是将tmp对象(即x的副本)的内容赋给了当前对象  }  // 返回当前对象的引用,以支持链式赋值操作  return *this;  
}

在这里插入图片描述

运行结果:

在这里插入图片描述

✍ 析构函数

有初始化资源的函数,就会有清理资源的函数。析构函数和构造函数一样,它是在当前作用域结束后就会自动调用析构函数。它的函数名字不一样,类名前面多了一个~

🏄 编译器默认生成的析构函数

一般情况下编译器也会默认生成一个析构函数,当我们的成员变量都没有申请资源时就不需要显示的写析构函数。

🏄 显式写的析构函数

	~Date(){year = 0;day = 0;month = 0;}

在这里插入图片描述

🏄 析构函数的特性

1、当前函数作用域结束后自动调用
2、无参数,无返回值
3、函数名是~+类名。
4、功能是清理对象中的资源而不是释放对象的空间。
5、支持显示调用,构造函数不支持。

在这里插入图片描述

🏄 没有深拷贝,导致二次释放同一空间问题

💘 问题的引入—拷贝构造函数

前面在讲拷贝构造函数和赋值构造函数,我们就对这个问题做了铺垫,这个问题的本质就和标题一样,内存重复释放,为什么会这样,本质还是万恶的值拷贝!下面我们写一段代码来解释并解决这个问题。

class Stack
{
public:Stack(){_capacity = 4;//假设开始的时候给容量设置为4_top = 0;_a = (int*)malloc(sizeof(int) * _capacity);if (_a == nullptr){std::cout << "malloc Failed" << std::endl;exit(-1);}}~Stack(){std::cout << " ~Stack" << std::endl;_capacity = 0;_top = 0;free(_a);_a = nullptr;}
private:int* _a;int _top;		// 栈顶int _capacity;  // 容量 
};int main()
{Stack st1;Stack st2(st1);return 0;
}

运行结果:

在这里插入图片描述

是的,程序在这里崩溃了,我们来调试一下。

在这里插入图片描述

可以看到st1中的_a、和st2中的_a保存的地址值是一模一样,释放了两次相同空间的地址。

在这里插入图片描述

💘 问题的解决—深拷贝

那么我们如何规避这种情况呢,就要用到深拷贝,我们可以开和st1中_a指向空间一样大小的数组,并把_a指向空间的值赋值给我们的数组。

在这里插入图片描述
代码实现:

// Stack类的拷贝构造函数  
// 接收一个对Stack类型的常量引用st作为参数,用于复制对象  
Stack(const Stack& st)  
{  // 为栈的底层数组_a分配内存,大小与源栈st的容量相同  _a = (int*)malloc(sizeof(int) * (st._capacity));  // 检查内存是否分配成功  if (_a == nullptr)  {  // 如果分配失败,则输出错误信息并退出程序  std::cout << "内存分配失败" << std::endl;  exit(-1);  }  // 使用memcpy函数将源栈st的底层数组内容复制到当前栈的底层数组_a中  memcpy(_a, st._a, sizeof(int) * (st._capacity));  // 复制源栈st的栈顶指针_top到当前栈  _top = st._top;  // 复制源栈st的容量_capacity到当前栈  _capacity = st._capacity;  
}

运行结果:

在这里插入图片描述

此时不再报错。

调试结果:

在这里插入图片描述
保存的地址不同(指向的空间不同),但是数组中的值相同,完成了拷贝构造(深拷贝)。

💘 赋值运算符重载函数的浅拷贝问题

使用编译器默认的值拷贝,去赋值,在刚刚的场景也会报错。

int main()
{Stack st1;Stack st2;st2 = st1;return 0;
}

在这里插入图片描述

我们可以使用赋值运算符重载函数现代写法来复用刚刚拷贝构造函数写好的深拷贝。

// Stack类的赋值运算符重载  
// 接收一个对Stack类型的常量引用st作为参数,用于赋值操作  
Stack& operator=(const Stack& st)  
{  // 检查是否自赋值,即当前对象与参数对象是否为同一个对象  if (this != &st)  {  free(_a);//释放之前_a的内存// 创建一个临时Stack对象tmp,并用参数对象st初始化  Stack tmp(st);  // 使用std::swap交换临时对象tmp的底层数组与当前对象的底层数组  std::swap(tmp._a, _a);  // 使用std::swap交换临时对象tmp的栈顶指针与当前对象的栈顶指针  std::swap(tmp._top, _top);  // 使用std::swap交换临时对象tmp的容量与当前对象的容量  std::swap(tmp._capacity, _capacity);  }  // 返回当前对象的引用,支持链式赋值操作  return *this;  
}

运行结果:

在这里插入图片描述

调试结果:

在这里插入图片描述
与预期一致。

✍ const成员函数

使用const关键字修饰的函数(放在函数括号右边)叫做const成员函数,const实际是修饰的this指针指向的内容,所以其指向的内容不能修改。

请看下面的代码:

class Date
{
public:Date(){year = 2024;month = 1;day = 1;}void f() const{this->year = 1;}
private:int year;int month;int day;
};int main()
{Date x;
}

在这里插入图片描述

🏄 const对象访问的规则

1、const对象不能访问非const类型的函数,但是可以构造函数和析构函数例外。这很好理解,因为我们的const修饰对象,对象的内容不能被修改,如果你能调用非const函数,在这个函数里修改了我们的成员变量,那不就逻辑不自洽了嘛。

在这里插入图片描述
const对象访问构造函数。

也不能对析构或者构造函数使用const修饰。

在这里插入图片描述

const对象访问析构函数。

在这里插入图片描述

const对象访问非const函数是非法的。

在这里插入图片描述

2、const成员函数类也不能调用其它的非const的成员函数。

在这里插入图片描述

本质上是一样的问题const Date* const this类型不能转变为Date* const this,否则它的能力就扩大了。

  • 注意前面的const修饰的是*this,表示this指向的内容具有常性,后面的const修饰this指针,表示this指针的值不能被修改。

🏄 非const对象访问的规则

3、但是非const成员函数内可以调用const成员函数。

在这里插入图片描述
程序正常退出。

4、非const对象也可以调用const函数。

在这里插入图片描述

3和4总结起来也是一样的,Date* const this类型可以向const Date* const this类型转化。

✍ 对普通对象的取地址运算符重载和对const对象取地址运算符重载

这两个函数一般都不需要我们显示的去写,编译器会默认生成的。

在这里插入图片描述

我们也可以显示的写出来。

// 非const成员函数的取地址运算符重载  
// 返回当前对象的地址  
Date* operator&()  
{  return this;  
}// const成员函数的取地址运算符重载  
// 返回当前对象的const地址  
const Date* operator&() const  
{  return this;  
}

但是通常只有我们有一些特殊的需求比如取出某个特定成员变量的地址时,才需要自己显示的去写。

✍ C++默认构造函数提供的机制

部分内容参考博主这篇博客C++默认构造函数提供的机制。

我们都知道,在C++98中,有着这样的几种构造函数:普通构造函数、析构函数、拷贝构造函数、赋值运算符重载函数。

生成这些特殊的成员函数(或者不生成)的规则比较复杂,编译器默认生成的构造函数是有可能被删除的。

🏄 C++默认构造函数是否提供的情况

  1. 如果自定义了普通构造函数和拷贝构造函数,系统将不再提供默认的无参构造函数。

在这里插入图片描述
但是如果定义了一个赋值运算符重载函数,系统还是会提供普通的无参构造函数。

在这里插入图片描述

2、如果自定义了一个普通的构造函数,系统还会提供一个拷贝构造函数和赋值运算符重载函数(值拷贝)。

  1. 如果自定义了一个拷贝构造函数,系统将不再提供默认的拷贝构造函数。但是会生成默认的赋值运算符重载函数。

在这里插入图片描述
4、如果自定义了一个赋值运算符重载函数,系统就不会默认生成赋值运算符重载函数了,但是其它函数还是会生成。

在这里插入图片描述
在这里插入图片描述
这里少打字了,应该是无参的构造函数。

没有生成报错是这样的:

在这里插入图片描述

  1. 如果自定义了一个析构函数,系统也就不会再提供默认的析构函数。

  2. 如果类里面有没有初始化的非静态const数据成员或者引用类型的数据成员,会导致默认提供的默认构造函数被删除。

在这里插入图片描述
当我们使用初始化列表初始化好这两个变量好,发现去调用拷贝构造函数是可以的(编译器默认生成了),但是拷贝赋值函数却被删除了。

在这里插入图片描述
在这里插入图片描述

6.用户如果自己没有提供一个拷贝构造函数或者拷贝赋值函数,编译器会隐式声明一个。

在这里插入图片描述

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

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

相关文章

在Windows系统上安装多个 Nodejs

前言 在Windows系统安装Nodejs 在Windows系统上安装多个 Nodejs v14.16.1安装位置 D:\sde\nodejs\node-v14.16.1-win-x64 v16.20.2安装位置 D:\sde\nodejs\node-v16.20.2-win-x64 v18.20.0安装位置 D:\sde\nodejs\node-v18.20.0-win-x64 v20.12.0安装位置 D:\sde\nod…

Java毕业设计-基于springboot开发的游戏分享网站平台-毕业论文+答辩PPT(附源代码+演示视频)

文章目录 前言一、毕设成果演示&#xff08;源代码在文末&#xff09;二、毕设摘要展示1、开发说明2、需求分析3、系统功能结构 三、系统实现展示1、系统功能模块2、后台登录2.1管理员功能模块2.2用户功能模块 四、毕设内容和源代码获取总结 Java毕业设计-基于springboot开发的…

ChatGLM2本地部署方法

chatglm2部署在本地时&#xff0c;需要从huggingface上下载模型的权重文件&#xff08;需要科学上网&#xff09;。下载后权重文件会自动保存在本地用户的文件夹上。但这样不利于分享&#xff0c;下面介绍如何将chatglm2模型打包部署。 一、克隆chatglm2部署 这个项目是chatgl…

“李子园”上榜中国民营企业社会责任优秀案例

日前&#xff0c;由浙江省工商联、浙江工商大学主办&#xff0c;杭州市工商联协办的2024浙江民营企业社会责任暨浙商ESG研讨会在杭州召开&#xff0c;探索民营企业履行社会责任的方法路径和趋势。会上公布了2023年中国民营企业社会责任优秀案例&#xff08;浙江入选企业&#x…

【小尘送书-第十五期】Excel函数与公式应用大全for Excel 365 Excel

大家好&#xff0c;我是小尘&#xff0c;欢迎你的关注&#xff01;大家可以一起交流学习&#xff01;欢迎大家在CSDN后台私信我&#xff01;一起讨论学习&#xff0c;讨论如何找到满意的工作&#xff01; &#x1f468;‍&#x1f4bb;博主主页&#xff1a;小尘要自信 &#x1…

【Linux】对进程地址空间的理解

一、关于进程地址空间的简单理解 进程地址空间其实是分了很多个区域的&#xff0c;区域划分的本质就是区域内的各个地址都是可以使用的。如同下面这个图所示&#xff1a; 无论是环境变量的地址还是环境变量表的地址&#xff0c;所存放的地址都在栈的上部。这里的已初始化数据和…

浅谈性能测试

本文主要针对WEB系统的性能测试。不涉及具体的执行操作&#xff0c;只是本人对性能测试的一点理解和认识。 性能测试的目的&#xff0c;简单说其实就是为了获取待测系统的响应时间、吞吐量、稳定性、容量等信息。而发现一些具体的性能相关的缺陷&#xff08;如内存溢出、并发处…

centos7系统下nginx1.24.0升级

如果没有这些目录&#xff0c;请先创建: mkdir /data/software mkdir /data/program提前下载所需的软件&#xff1a; cd /data/software wget https://ftp.pcre.org/pub/pcre/pcre-8.42.tar.gz wget https://nginx.org/download/nginx-1.24.0.tar.gz安装nginx cd /data/soft…

微信小程序开发之常用组件解释

1 基础内容组件 1.1text组件 text的功能主要是用于内联文本&#xff0c;与网页中的span有点类似。 主要属性有 例子&#xff1a;页面上添加一个可以选中的文本 在wxml文件中添加&#xff1a; <view> <text user-select>17544456565</text> </view>…

洗地机哪个好?专业对比,帮你选出比较适合的洗地机

随着科技的不断发展&#xff0c;洗地机已经成为了现代生活中不可或缺的清洁工具。然而&#xff0c;市面上涌现出各种各样的洗地机品牌&#xff0c;品质良莠不齐。因此&#xff0c;选择一个可靠的品牌至关重要&#xff0c;以确保产品质量和使用效果。为了帮助大家更好地选择&…

数据结构 之 栈与单调栈习题 力扣oj(附加思路版)

#include<stack> --栈的头文件 栈的特点 &#xff1a; 先进后出 &#xff0c; 后进先出 相关函数&#xff1a; top() 获取栈顶元素 ,返回栈顶元素的值 pop() 删除栈顶元素 ,没有返回值 push() 放入元素 ,没有返回值 empty() 为空返回 true 否则返回false size() 元素…

二叉树|701.二叉搜索树中的插入操作

力扣题目链接 class Solution { public:TreeNode* insertIntoBST(TreeNode* root, int val) {if (root NULL) {TreeNode* node new TreeNode(val);return node;}if (root->val > val) root->left insertIntoBST(root->left, val);if (root->val < val) r…

从后端到前端

原文地址&#xff1a;从后端到前端 - Pleasure的博客 下面是正文内容&#xff1a; 前言 在前面几章中主要介绍了系统开发的后端部分&#xff0c;但是验证接口的适用性只能通过专门的软件&#xff08;Apifox&#xff0c;Postman等&#xff09;来进行测试。那从现在开始&#xf…

【Java程序设计】【C00376】基于(JavaWeb)Springboot的社区帮扶对象管理系统(有论文)

【C00376】基于&#xff08;JavaWeb&#xff09;Springboot的社区帮扶对象管理系统&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;已经做了六年的毕业设计程序开发&am…

element UI季度选择器的实现

效果展示 用elementUI的select实现季度选择器 代码实现 generateQuarterOption放在methods中&#xff0c;需要近几年的只需要修改第一个循环的次数即可&#xff0c;mounted生命周期函数中调用generateQuarterOption() generateQuarterOption() {//近3年所有季度let now ne…

记录一次使用cert-manager-颁发CA证书

一、官网 SelfSigned - cert-manager Documentation 二、例子 apiVersion: v1 kind: Namespace metadata:name: sandbox --- apiVersion: cert-manager.io/v1 kind: ClusterIssuer metadata:name: selfsigned-issuer spec:selfSigned: {} --- apiVersion: cert-manager.io/v…

如何删除Excel中的空白行?这里提供详细步骤

要从数据集中删除所有空白行吗&#xff1f;如果是这样&#xff0c;Microsoft Excel提供自动和手动方法来清除空白行并向上移动数据。下面是如何使用这些方法。 删除空白行时&#xff0c;Excel会删除整行并上移数据&#xff0c;以便数据集中不再有空行。记住&#xff0c;你也可…

1.7.2 练习

一、projecrion函数 题目&#xff1a;projecrion函数中的第一个参数fov和第二个参数ratio参数进行实验。看能否搞懂它们是如何影响透视平截头体的。 当ratio不变&#xff0c;fov值变大&#xff0c;显示的物体会变小&#xff1b;当fov不变&#xff0c;radio值变大&#xff0c;…

机器学习:探索数据中的模式与智能

文章目录 导言介绍&#xff1a;机器学习的定义和重要性发展历程&#xff1a;从概念到现实应用 基础概念机器学习的基本原理监督学习、无监督学习和强化学习的区别与应用1.监督学习2.无监督学习3.强化学习 常见的机器学习任务和应用领域 结语 导言 当代科技领域中最为引人注目的…

git reset版本回退后悔药(图文例子)

目录 版本回退前期测试样例准备git reset --soft 不撤销add,撤销commit,保留修改git reset --mixed 或 git reset () 撤销add,撤销commit,保存修改git reset --hard 撤销add,撤销commit,不保存修改git reset --merge 取消合并git reset --keep 不撤销add,撤销commit,根据情况判…