【C++】类与对象 第二篇(构造函数,析构函数,拷贝构造,赋值重载)

news2024/11/23 20:27:08

目录

类的6个默认成员函数

初始化和清理

1.构造函数

2.析构函数

3.共同点

拷贝复制

1.拷贝构造

使用细节

2.赋值重载

运算符重载

== <= < >= > !=

连续赋值


C++入门 第一篇(C++关键字, 命名空间,C++输入&输出)-CSDN博客

C++入门 第二篇( 引用、内联函数、auto关键字、指针空值nullptr)-CSDN博客

【C++】类与对象 第一篇(class,this)-CSDN博客

类的6个默认成员函数

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

初始化和清理

1.构造函数

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

  1. 函数名与类名相同。

  2. 无返回值。

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

  4. 构造函数可以重载。

 class Stack
 {
 public:
     Stack()
     {
         _a=nullptr;
         _size=_capacity=0;
     }
 ​
     Stack(int n)
     {
         _a=(int*)malloc(sizeof(int)*n);
         _size=_capacity=0;
     }
 ​
     void Init(int n=4)
     {
         _a=(int*)malloc(sizeof(int)*n);
         if (nullptr==_a)
         {
             perror("malloc is fail");
             return;
         }
         _capacity=n;
         _size=0;
     }
 ​
     void Push(int x)
     {
         //...
         a[_size++]=x;
     }
     //...
     void Dstory()
     {
         //...
     }
 ​
 private:
     int _a;
     int _size;
     int _capacity;
 };
 ​
 int main()
 {
     Stack st;//无参
    //Stack st();//有参
 ​
     st.Push(1);
     st.Push(2);
     st.Push(3);
     st.Push(4);
 ​
     st.Dstory();
 ​
     return 0;
 }

以上述代码为例:

自动调用初始化

注意:调用无参时如:Data d;此处在后面不能➕(),否则编译器会调用有参的。

使用缺省值:

 class Date
 {
 public:
     Date(int year=1,int month=1,int day=1)
     {
         _year=1;
         _month=1;
         _day=1;
     }
 ​
     void print()
     {
         cout<<_year<<"/"<<_month<<"/"<<_day<<endl;
     }
 ​
 private:
     //成员变量
     int _year;
     int _month;
     int _day;
 };
 ​
 int main()
 {
     //Date d1;
     Date d2(2077,2,3);
     d2.print();
 ​
     return 0;
 }

2.析构函数

概念: 通过前面构造函数的学习,我们知道一个对象是怎么来的,那一个对象又是怎么没呢的? 析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由 编译器完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作

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

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

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

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

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

例子:

 typedef int DataType;
 class Stack
 {
 public:
     Stack(size_t capacity = 3)
     {
         _array = (DataType *)malloc(sizeof(DataType) * capacity);
         if (NULL == _array)
         {
             perror("malloc申请空间失败!!!");
             return;
         }
         _capacity = capacity;
         _size = 0;
     }
     void Push(DataType data)
     {
         // CheckCapacity();
         _array[_size] = data;
         _size++;
     }
     // 其他方法...
     ~Stack()
     {
         if (_array)
         {
             free(_array);
             _array = NULL;
             _capacity = 0;
             _size = 0;
         }
     }
 ​
 private:
     DataType *_array;
     int _capacity;
     int _size;
 };
 void TestStack()
 {
     Stack s;
     s.Push(1);
     s.Push(2);
 }

3.共同点

如果编译过程不写,那编译器会自动生成一个默认的,但是如果我们实现了任意一个,编译器就不会生成了。

若是自动初始化,那为什么这个地方会生成随机值呢?

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

内置类型(基本类型):int/char/double... /任意类型指针 自定义类型:class/structd定义的

默认生成构造函数:

1.内置类型成员不做处理。

2.自定义类型的成员,会去调用它的默认构造(不用传参数的构造)

 private:
     // 基本类型/内置类型 - 不进初始化
     int _year;
     int _month;
     int _day;

实用场景:

 class Date
 {
 public:
     //内置类型成员不做处理
     void print()
     {
         cout<<_year<<"/"<<_month<<"/"<<_day<<endl;
     }
 ​
 private:
     // 基本类型/内置类型 - 不进初始化
     int _year;
     int _month;
     int _day;
 };
 ​
 class MyQueue
 {
     // 默认生成构造函数,对自定义类型,会调用它的默认构造函数
     void push(int x);
     {
 ​
     }
     //...
   
     Stack _pushST();
     Stack _popST();
 };

析构:

默认生成构造函数:

1.内置类型成员不做处理。

2.自定义类型的成员,会去调用它的析构函数

 class MyQueue
 {
     // 默认生成析构函数,对自定义类型,会调用它的析构函数
     void push(int x);
     {
 ​
     }
     Stack _pushST();
     Stack _popST();
 };
 ​
 int main()
 {
     //Date d1;
     Date d1;
     d1.print();
 ​
     MyQueue q;
 ​
     return 0;
 }

内置类型并不会主动初始化,但可以通过给缺省值进行初始化:

 private:
 //声明位置给缺省值
     int _year=1;
     int _month=1;
     int _day=1;

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

拷贝复制

1.拷贝构造

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

拷贝构造:内置类型,编译器直接拷贝,自定义类型拷贝需要调用拷贝构造

特征

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

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

    为什么自定义类型要用拷贝构造,而内置类型编译器却可以直接拷贝(按字节拷贝)? ​ 因为自定义类型进行拷贝容易出问题:若要拷贝栈,栈的成员变量都是指向相同的空间,若进行拷贝,则会导致两个Stack指向同一个空间,导致原本Stack执行析构函数时被销毁,而复制的那个则指向空,正确的应该是各有各的空间。

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

class Date
 {
 public:
     Date(int year=2077, int month=10, int day=12)
     {
         _year = year;
         _month = month;
         _day = day;
     }
     Date(Date d)//这样拷贝会造成无限递归
     {
         _year = d.year;
         _month = d.month;
         _day = d.day;
     }
 ​
 private:
     int _year;
     int _month;
     int _day;
 };

在拷贝构造函数 Date(Date d) 中,参数 d 是按值传递的,这意味着每次调用拷贝构造函数时都会创建一个新的 Date 对象,并将原始对象 d 复制到新的对象中。然而,在拷贝构造函数内部,对于拷贝构造函数的调用又会传递同样的参数 d,导致不断地递归调用拷贝构造函数,从而产生无限递归。

则可以使用 & ,使其不在重新创建空间并使用原Date对象

Date(const Date& d)
 {
     _year = d._year;
     _month = d._month;
     _day = d._day;
 }

这个拷贝构造函数将创建一个新的 Date 对象,并将原始对象的 _year_month_day 成员变量的值分别拷贝到新对象的相应成员变量中。通过使用对象引用作为参数,我们可以避免无限递归调用拷贝构造函数的问题,并且确保在构造新对象时不会复制整个对象

 
class Date
 {
 public:
     Date(int year = 2077, int month = 10, int day = 12)
     {
         _year = year;
         _month = month;
         _day = day;
     }
 ​
     Date(const Date& d)//const是为了防止写反->d._year = _year;加const缩小权限
     {
         cout<<"Date(Date& d);"<<endl;
         _year = d._year;
         _month = d._month;
         _day = d._day;
     }
 ​
 private:
     int _year;
     int _month;
     int _day;
 };
 ​
 int main()
 {
     Date d1(2023, 2, 3);
     //两种拷贝方式
     Date d2(d1); // 使用拷贝构造函数创建对象 d2,并将 d1 的值拷贝到 d2 中
     Date d3=d1;
 ​
     return 0;
 }

RunCode:

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

深拷贝是指在对象拷贝时,复制所有的数据和资源,使得新对象和原对象完全独立,互不影响。与之相对的是浅拷贝,浅拷贝只复制指针或引用,导致新旧对象共享同一份数据,修改一个可能会影响另一个。深拷贝能够保证对象之间的独立性和数据完整性。

分清楚是不是拷贝构造:

 class Date
 {
 public:
     Date(int year = 2077, int month = 10, int day = 12)
     {
         _year = year;
         _month = month;
         _day = day;
     }
   
     //拷贝构造函数   
     Date(const Date& d)
     {
         cout<<"Date(Date& d);"<<endl;
         _year = d._year;
         _month = d._month;
         _day = d._day;
     }
 ​
     //构造函数 不是拷贝构造
     Date(const Date* d)
     {
         cout<<"Date(Date& d);"<<endl;
         _year = d._year;
         _month = d._month;
         _day = d._day;
     }
 private:
     int _year;
     int _month;
     int _day;
 };
 ​
 int main()
 {
     Date d1(2023, 2, 3);
 //Date(const Date* d)
     Date d2(&d1); 
     Date d3=&d1;
 ​
     return 0;
 }

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

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

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

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

使用细节

1使用拷贝构造时最好加上const,以防权限的放大

   // 拷贝构造
     Date(const Date &d)
     {
         _year = d.year;
         _month = d.month;
         _year = d.year;
     }

2编译器可自动生成但存在问题

本文所用的日期类可以使用自动生成拷贝,但进行拷贝栈时就会出现问题

当进行拷贝时,用值拷贝,导致两个栈指向同一个空间,造成空间互相覆盖 指向同一块空间的问题:1.插入删除数据会互相影响 2.析构两次,程序崩溃

析构是也跟栈一样是后进先出后定义的先析构,故st2先析构,对st2滞空是,并不影响st1,导致野指针。此时浅拷贝行不通,需要进行深拷贝:

代码实现:

class Stack
 {
 public:
     void Push(const int &data)
     {
         _array[_size] = data;
         _size++;
     }
 ​
     Stack(const Stack &st)
     {
         _array = (int *)malloc(sizeof(int) * st._capacity);
         if (nullptr == _array)
         {
             perror("malloc is fail");
             exit(-1);
         }
         memcpy(_array, st._array, sizeof(int) * st._size);
         _size = st._size;
         _capacity = st._capacity;
     }
 ​
     ~Stack()
     {
         if (_array)
         {
             delete[] _array;
             _array = nullptr;
             _capacity = 0;
             _size = 0;
         }
     }
 ​
 private:
     int *_array;
     int _size;
     int _capacity;
 };
 ​
 int main()
 {
     Stack st1;
     st1.Push(1);
     st1.Push(2);
     st1.Push(3);
 ​
     Stack st2(st1);
 ​
     return 0;
 }

什么情况下需要事项拷贝构造呢? 自己实现了析构释放空间,就需要实现拷贝构造

 class Stack
 {
 public:
     //构造函数  
     Stack(size_t _capacity = 10)
     {
         cout << "Stack(size_t _capacity)" << endl;
         _array = (int *)malloc(sizeof(int) * _capacity);
         if (nullptr == _array)
         {
             perror("malloc is fail");
             exit(-1);
         }
         _size = 0;
         _capacity = _capacity;
     }
 ​
     void Push(const int &data)
     {
         _array[_size] = data;
         _size++;
     }
 ​
     //拷贝构造
     Stack(const Stack &st)
     {
         _array = (int *)malloc(sizeof(int) * st._capacity);
         if (nullptr == _array)
         {
             perror("malloc is fail");
             exit(-1);
         }
         memcpy(_array, st._array, sizeof(int) * st._size);
         _size = st._size;
         _capacity = st._capacity;
     }
 ​
     ~Stack()
     {
         if (_array)
         {
             delete[] _array;
             _array = nullptr;
             _capacity = 0;
             _size = 0;
         }
     }
 ​
 private:
     int *_array;
     int _size;
     int _capacity;
 };
 ​
 class MyQueue
 {
     //默认生成构造 ->生成定义在Stack中的默认构造函数
     //默认生成析构
     //默认生成的拷贝构造
     
 private:
     Stack _pushST;
     Stack _popST;
     int _size = 0;
 };
 ​
 int main()
 {
     Stack st1;
     st1.Push(1);
     st1.Push(2);
     st1.Push(3);
 ​
     Stack st2(st1);
     cout<<"====="<<endl;
     MyQueue q;
 ​
     return 0;
 }

默认生成拷贝构造和赋值重载:

a.内置类型完成 浅/值 拷贝--按byte一个一个拷贝

b.自定义类型,去调用这个成员 拷贝构造/赋值重载

2.赋值重载

运算符重载

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

函数名字为:关键字operator后面接需要重载的运算符符号。 函数原型:返回值类型 operator操作符(参数列表)

自定义类型不能直接使用运算操作符 内置类型是语法定义,而自定义类型为人为定义,编译器并不知道该如何进行比较,这便出现了operator用函数完成

运算符重载:自定义类型对象可以使用运算符

函数重载:支持函数名相同,参数不同的函数

 
#include <iostream>
 using namespace std;
 ​
 class Date
 {
 public:
     // 构造函数
     Date(int year = 1, 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;
     }
 ​
     void Print()
     {
         cout << _year << "/" << _month << "/" << _day << endl;
     }
 ​
     // d1==d2 -> d1.operator==(d2)
     bool operator==(const Date &d)
     {
         return _year == d._year && _month == d._month && _day == d._day;
         //    this->_year==d.year
     }
 ​
 private:
     // 成员变量
     int _year;
     int _month;
     int _day;
 };
 ​
 int main()
 {
     Date d1(2077, 2, 4);
     Date d2(2077, 2, 4);
 ​
     // d1==d2;
     cout << d1.operator==(d2) << endl;
     cout << (d1 == d2) << endl;
 ​
     return 0;
 }

注意:

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

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

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

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

  • 注意以下5个运算符不能重载

     .*   ::  sizeof  ?:   .

== <= < >= > !=
// d1==d2 -> d1.operator==(d2)
     bool operator==(const Date &d)
     {
         return _year == d._year && _month == d._month && _day == d._day;
         //    this->_year==d.year
     }
 ​
     // 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;
     // }
 ​
     // d1<d2
     bool operator<(const Date &d)
     {
         return _year < d._year && (_year == d._year && 
        _month < d._month) && (_year == d._year && 
        _month == d._month && _day < d._day);
     }

任何一个类都适用于,当写了一个>=或<=那其他的就可以进行复用

     // d1<=d2
     bool operator<=(const Date &d)
     {
         return *this < d || *this == d;//*this就是d1
     }

这个地方对operator<(const Date &d)进行了复用。operator<=(const Date &d)中的表达式*this < d调用了operator<(const Date &d)来判断两个日期对象的大小关系。同时,operator<=(const Date &d)还利用operator==(const Date &d)来判断两个日期对象是否相等。通过这样的复用方式,可以简化代码并提高代码的可读性

     // d1>d2
     bool operator>(const Date &d)
     {
         return !(*this<= d);
     }
 ​
     //d1>=d2
     bool operator>=(const Date& d)
     {
         return !(*this<d);
     }
 ​
     //d1!=d2
     bool operator!=(const Date& d)
     {
         return !(*this ==d);
     }

当然赋值运算符不使用&并不会无限循环

    // d1<=d2
     bool operator<=(const Date d)
     {
         return *this < d || *this == d;
     }

当然可以不用,但是最好还是加上

连续赋值

d3 = d1 = d2

     // 因为Date operator=(const Date &d) 出了该函数的作用域*this/d1 还在此时返回的*this是临时拷贝,需要进行拷贝重新开空间造成浪费,不如直接进行&
     Date operator=(const Date &d)//&d 引用
     {
         if (this != &d)//防止d1=d1进行赋值 此处的&为去地址 ,若this与d的地址一样则无需赋值
         {
             _year = d._year;
             _month = d._month;
             _day = d._day;
         }
 ​
         return *this; //*this就是d1
     }                 // 返回值为了支持连续赋值,保持运算符特性  d3=d1=d2;

当然要注意,赋值重载是针对已经创造的对象

 Date d5=d1;//拷贝构造

像这样就是拷贝构造

同样作为默认成员函数,可以不用手动写

源码:

 #include <iostream>
 using namespace std;
 ​
 class Date
 {
 public:
     // 构造函数
     Date(int year = 1, 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;
     }
 ​
     void Print()
     {
         cout << _year << "/" << _month << "/" << _day << endl;
     }
   
     // d1==d2 -> d1.operator==(d2)
     bool operator==(const Date &d)
     {
         return _year == d._year && _month == d._month && _day == d._day;
         //    this->_year==d.year
     }
 ​
     // d1<d2
     bool operator<(const Date &d)
     {
         return _year < d._year && (_year == d._year && _month < d._month) && (_year == d._year && _month == d._month && _day < d._day);
     }
 ​
     // d1<=d2
     bool operator<=(const Date &d)
     {
         return *this < d || *this == d; //*this就是d1
     }
 ​
     // d1>d2
     bool operator>(const Date &d)
     {
         return !(*this <= d);
     }
 ​
     // d1>=d2
     bool operator>=(const Date &d)
     {
         return !(*this < d);
     }
 ​
     // d1!=d2
     bool operator!=(const Date &d)
     {
         return !(*this == d);
     }
 ​
     // // 因为Date operator=(const Date &d) 出了该函数的作用域*this/d1 还在此时返回的*this是临时拷贝,需要进行拷贝重新开空间造成浪费,不如直接进行&
     // Date operator=(const Date &d)//&d 引用
     // {
     //     if (this != &d)//防止d1=d1进行赋值 此处的&为去地址 ,若this与d的地址一样则无需赋值
     //     {
     //         _year = d._year;
     //         _month = d._month;
     //         _day = d._day;
     //     }
 ​
     //     return *this; //*this就是d1
     // }                 // 返回值为了支持连续赋值,保持运算符特性  d3=d1=d2;
 ​
 private:
     // 成员变量
     int _year;
     int _month;
     int _day;
 };
 ​
 int main()
 {
     Date d1(2077, 2, 4);
     Date d2(2222, 3, 2);
     Date d3(2078, 2, 4);
 ​
     cout << (d1 < d2) << endl;
     // d3 = d1 = d2;
 ​
     d1 = d2;
     d1.Print();
 ​
     return 0;
 }

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

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

相关文章

036-第三代软件开发-系统时间设置

第三代软件开发-系统时间设置 文章目录 第三代软件开发-系统时间设置项目介绍系统时间设置演示效果QML 实现小伙伴自创 TumblerQt 家 Tumbler C 端实现 总结一下 关键字&#xff1a; Qt、 Qml、 Time、 时间、 系统 项目介绍 欢迎来到我们的 QML & C 项目&#xff01;…

用Python实现感知机学习算法及其对偶算法实验报告

实验目的 1.理解感知机学习算法的基本思想&#xff1a;感知机是一种简单的线性分类模型&#xff0c;其基本思想是通过不断调整权重&#xff0c;使得分类超平面能够将不同类别的样本正确分开。 2.掌握感知机学习算法的实现方法&#xff1a;感知机学习算法基于随机梯度下降法&am…

【斗罗二】霍雨浩迷惑审查,戴华斌故意挑衅,惨败者屈服下跪

【侵权联系删除】【文/郑尔巴金】 深度爆料&#xff0c;自《绝世唐门》宣布问世以来&#xff0c;其在国漫圈引发的关注和热议便如火如荼。作为《斗罗大陆》的续作&#xff0c;这部作品无疑继承了前作的荣光&#xff0c;甚至被无数粉丝期待着能再创辉煌。在各大社交媒体和国漫论…

前端移动web高级详细解析二

移动 Web 第二天 01-空间转换 空间转换简介 空间&#xff1a;是从坐标轴角度定义的 X 、Y 和 Z 三条坐标轴构成了一个立体空间&#xff0c;Z 轴位置与视线方向相同。 空间转换也叫 3D转换 属性&#xff1a;transform 平移 transform: translate3d(x, y, z); transform…

搞定蓝牙——第五篇(SMP)

搞定蓝牙——第五篇&#xff08;SMP&#xff09; 原理Security Manager(简称SM&#xff0c;不要想歪)秘钥配对秘钥生成特定秘钥分发原理总结 原理 Security Manager(简称SM&#xff0c;不要想歪) 按照前面的试验&#xff0c;两个设备可以通过ble通讯了&#xff0c;但是&#…

(a /b)*c的值

系列文章目录 进阶的卡莎C++_睡觉觉觉得的博客-CSDN博客数1的个数_睡觉觉觉得的博客-CSDN博客双精度浮点数的输入输出_睡觉觉觉得的博客-CSDN博客足球联赛积分_睡觉觉觉得的博客-CSDN博客大减价(一级)_睡觉觉觉得的博客-CSDN博客小写字母的判断_睡觉觉觉得的博客-CSDN博客纸币(…

ETL工具Kettle

1 Kettle的基本概念 一个数据抽取过程&#xff0c;主要包括创建一个作业&#xff08;Job&#xff09;&#xff0c;每个作业由一个或多个作业项&#xff08;Job Entry&#xff09;和连接作业项的作业跳&#xff08;Job Hop&#xff09;组成。每个作业项可以是一个转换&#xff…

Cookie技术

Cookie中文名称为小型文本文件&#xff0c;指某些网站为了辨别用户身份、进行会话跟踪而储存在用户本地终端上的数据。 Cookie是由服务器端生成&#xff0c;发送给User-Agent&#xff08;—般是浏览器&#xff09;&#xff0c;浏览器会将Cookie的key/value保存到某个目录下的文…

wiresharak捕获DNS

DNS解析&#xff1a; 过滤项输入dns&#xff1a; dns查询报文 应答报文&#xff1a; 事务id相同&#xff0c;flag里 QR字段1&#xff0c;表示响应&#xff0c;answers rrs变成了2. 并且响应报文多了Answers 再具体一点&#xff0c;得到解析出的ip地址&#xff08;最底下的add…

【Linux】虚拟机安装Linux、客户端工具,MobaXterm的使用,Linux常用命令

目录 一&#xff0c;安装Linux的centos7版本 具体安装步骤&#xff1a; 二&#xff0c;Linux常见的命令&#xff1a; 三、安装客户端工具 1、介绍 2、安装MobaXterm 3、换源 四、拍照功能 一&#xff0c;安装Linux的centos7版本 介绍&#xff1a; 具体安装步骤&#…

PTA L1-8 静静的推荐

PTA L1-8 静静的推荐 分数 20 全屏浏览题目 切换布局 作者 陈越 单位 浙江大学 天梯赛结束后&#xff0c;某企业的人力资源部希望组委会能推荐一批优秀的学生&#xff0c;这个整理推荐名单的任务就由静静姐负责。企业接受推荐的流程是这样的&#xff1a; 只考虑得分不低于 175 …

软考系列(系统架构师)- 2012年系统架构师软考案例分析考点

试题一 软件架构&#xff08;架构风格对比、架构风格选取、架构设计过程&#xff09; 【问题1】&#xff08;12分&#xff09; 请用200字以内的文字解释什么是软件架构风格&#xff0c;并从集成开发环境与用户的交互方式、集成开发环境的扩展性、集成开发环境的数据管理三个方…

Anaconda下载和安装

1.概述 1&#xff09;包含conda&#xff1a;conda是一个环境管理器&#xff0c;其功能依靠conda包来实现&#xff0c;该环境管理器与pip类似。 2&#xff09;安装大量工具包&#xff1a;Anaconda会自动安装一个基本的python&#xff0c;该python的版本Anaconda的版本有关。该…

进程(详解)

进程 进程PCB进程的定义进程的组成进程模式进程的状态进程的运行进程的创建进程的结束孤儿进程僵尸进程僵尸进程的危害 进程的创建pidforkwait案例 进程 PCB 从操作系统理解进程概念-------先描述&#xff0c;后组织 为了使参与并发执行的程序能独立的运行&#xff0c;必须为之…

经常遇到的问题

一个前端经常会遇到的问题 例如&#xff0c;我想要在一个项目里&#xff0c;监听所有的fetch请求&#xff0c;应该怎么办&#xff1f;又或者说&#xff0c;我想用别人封装好的方法&#xff0c;但是在它之前&#xff0c;需要经过一层处理、判断&#xff0c;然后再看情况是否调用…

如何隐藏woocommerce 后台header,woocommerce-layout__header

如何隐藏woocommerce 后台header&#xff0c;woocommerce-layout__header WooCommerce |Products Store Activity| Inbox| Orders| Stock| Reviews| Notices| breadcrumbs 在 functions.php 里添加如下代码即可&#xff1a; // Disable WooCommerce Header in WordPress Admi…

开启CETOS 裸奔了一年的服务器开启firewall防火墙

记录一下关于firewall&#xff0c;博主非运维专家或服务器专家。 背景 客户有一台裸奔运行了一年多的系统有公网但发现没有开防火墙&#xff0c;iptables和firewall均是关闭状态&#xff0c;通过扫描发现很多漏洞。根据客户要求对端口进行重新梳理且关闭不必要或有潜在风险的…

kubeadmin部署k8s1.27.4

kubeadmin部署k8s1.27.4 环境介绍 IP主机名资源配置系统版本192.168.117.170k8s-master2c2g200gCentos7.9192.168.117.171k8s-node12c2g200gCentos7.9192.168.117.172k8s-node22c2g200gCentos7.9 编辑本地解析且修改主机名 三台主机都要做 vim /etc/hosts配置主机名 mast…

软考系列(系统架构师)- 2013年系统架构师软考案例分析考点

试题一 软件架构&#xff08;根据描述填表、ESB 定义和功能&#xff09; 【问题1】&#xff08;10分&#xff09; 服务建模是对Ramp Coordination信息系统进行集成的首要工作&#xff0c;公司的架构师首先对Ramp Coordination信息系统进行服务建模&#xff0c;识别出系统中的两…

lesson2(补充)关于>>运算符和<<运算符重载

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 前言&#xff1a; cout和cin我们在使用时需要包含iostream头文件&#xff0c;我们可以知道的是cout是写在ostream类里的&#xff0c;cin是写在istream类里的&#xff0c;他们都是定义出的对象&#xff0c;而<< 和 >…