目录
类的默认成员函数
类中共有6个默认成员函数,即自己不实现,编译器会帮助我们实现出来。构造函数、析构函数、拷贝构造、赋值运算符重载、const成员、取地址及const取地址操作符重载。
一. 构造函数
1. 概念
在对象构造时调用的函数,这个函数完成初始化工作
2.无参时主函数中的写法
#include<iostream>
using namespace std;
class date
{
public:
//没有返回值
date(int year, int month, int day)//构造函数名与类名相同
{
_year = year;
_month = month;
_day = day;
}
date()//构造函数可以重载
{
_year = 0;
_month = 0;
_day = 0;
}
void print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
//构造函数无参时
//date d();错误写法会报错
date d;//正确写法
d.print();
return 0;
}
构造函数无参是一种特殊情况,不能像普通成员函数一样写括号,只写类的实例化即可。编译器无法区分 date d(); 是函数的声明还是定义对象
3. 特性
特性1-3
1.没有返回值
2.函数名跟类名相同
3.对象实例化时编译器自动调用对应的构造函数
#include<iostream>
using namespace std;
class date
{
public:
//没有返回值
date(int year,in tmonth,int day)//构造函数名与类名相同
{
_year=year;
_month=month;
_day=day;
}
private:
int _year;
int _month;
int _day;
int main()
{
date d(2023,2,7);//类的对象实例化自动调用构造函数
return 0;
}
特性 4
4.构造函数可以重载(一个类有多个构造函数)
class date
{
public:
//没有返回值
date(int year,in tmonth,int day)//构造函数名与类名相同
{
_year=year;
_month=month;
_day=day;
}
date ()//构造函数可以重载
{
....
}
private:
int _year;
int _month;
int _day;
int main()
{
date d(2023,2,7);//类的对象实例化调用构造函数
return 0;
}
特性 5
如果类中没有显式定义构造函数,则c++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成
#include<iostream>
using namespace std;
class date
{
public:
//编译器自动生成默认构造函数
void print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d;
d.print();//-858993460--858993460--858993460
return 0;
}
若输出结果,则会发现为随机值对于编译器自动生成的默认构造函数, 针对内置类型的成员变量没有做处理。
自定义类型
#include<iostream>
using namespace std;
class Time
{
public:
Time()//默认构造函数
{
_hours = 0;
_minute = 0;
_seconds = 0;
}
private:
int _hours;
int _minute;
int _seconds;
};
class date
{
public:
//没有构造函数,则编译器会自动生成一个无参的默认构造函数
void print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
Time _t;//调用自定义类型
};
int main()
{
date d;//无参数
d.print();
return 0;
}
特性 6
无参的构造函数和全缺省的构造函数都被称为默认构造函数,并且默认构造函数只能有一个对于无参的,两者都可以使用,就没必要共同存在默认构造函数:(不用传参数)
1.自己实现的无参的构造函数
2.自己实现的全缺省构造函数
3.自己没写编译器自动生成的
- 既想要带参数,又想要不带参数的 如何使用一个构造函数完成?
全缺省若参数没有传过去,则使用缺省参数。若有参数,则直接进入函数中。
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d;//无参数
d.print();//1-1-1
date d2(2022, 12, 20);//带参数
d2.print();//2022-12-20
return 0;
}
二.析构函数
1. 概念
对象在销毁时会自动调用析构函数,完成类的一些资源清理工作
2. 特性
析构函数名是在类名前加上字符~ (~在c语言表示按位取反)
无参数无返回值类型。
一个类只能由一个析构函数,若未显式定义,系统会自动生成默认的析构函数(析构函数不能重载)。
对象生命周期结束,C++编译系统自动调用析构函数。
#include<iostream>
using namespace std;
class date
{
public:
date()
{
}
//没有返回值并且无参
~date()//析构函数,若我们自己不写,系统会自动生成一个析构函数
{
}
int main()
{
date d;
return 0;//调试时,走到return 0这步 F11会进入析构函数
}
三.拷贝构造函数
1.值传递
#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(date d)//值传递 date d 会报错造成无限循环
{
_year = d._year;
_month = d._month;
_day = d._day;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(d1);//拷贝构造
return 0;
}
2.浅拷贝问题
#include<iostream>
using namespace std;
class stack
{
public:
stack(int n)//构造函数
{
_a = (int*)malloc(sizeof(int) * n);
_size = 0;
_capity = n;
}
~stack()//析构函数
{
free(_a);
_a = nullptr;
_size = _capity = 0;
}
private:
int * _a;
int _size;
int _capity;
};
int main()
{
stack s1(10);
stack s2(s1);//拷贝构造
return 0;//空间会被释放两次,程序崩溃
}
四.赋值运算符重载
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
return 0;
}
date 实例化出的对象 d1 与d2 的大小可以直接比较嘛?
不可以,内置类型是编译器自己定义的类型,它知道要怎么比
自定义类型是自己定义的,怎么比较大小由自己规定。C++为了增强代码的可读性引入运算符重载,运算符重载是具有特殊函数名的函数 ,运算符重载就是为了给自定义类型进行比较等提供的函数。
函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型 operator操作符(参数列表)。
1. 操作符 ==
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
bool operator ==( const date& d2)// 由于隐藏的this指针的存在,&d1传过来由this指针接收
{
return _year == d2._year && _month == d2._month && _day == d2._day;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
d1 == d2;// d1.operator==(d2) 等价于 d1.operator==(&d1,d2)
cout << (d1 == d2) << endl;
return 0;
}
传入类中,由于隐藏的this指针的存在,取第一个参数d1的地址传过去被this指针接收,_year等价于d1._year。但是由于this指针是隐藏的,所以&d1也不需要表现出来直接传入d2即可
注意事项
1.不能通过连接其他符号来创建新的操作符 (如 operator@)
2.重载操作符必须有一个类类型参数
3.用于内置类型的操作符,其含义不能改变(如 int 加法 不能改变)
这里理解很简单,因为运算符重载是针对自定义类型的,对于内置类型,编译器都已经设置好了
4.作为类成员的重载函数时,其形参看起来比操作数数目少1成员函数的操作符有一个默认的形参this,限定为第一个形参
2. 操作符 >
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
bool operator >( const date& d2)// 由于隐藏的this指针的存在,&d1传过来由this指针接收
{
if (_year > d2._year)//年大为大
{
return true;
}
else if (_year == d2._year && _month > d2._month) //年相等,月大为大
{
return true;
}
else if (_year == d2._year && _month == d2._month && _day > d2._day)//年 月相等,天大为大
{
return true;
}
else
{
return false;
}
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
d1 > d2;//等价于 operator>(&d1,d2)
cout << (d1 > d2) << endl;
return 0;
}
3. 操作符 !=
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
bool operator==(const date& d2)//d1==d2
{
return _year == d2._year && _month == d2._month && _day == d2._day;
}
bool operator!=(const date& d2)
{
return !(*this == d2);//借助上面的d1==d2的相反即 d1!=d2
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
d1 != d2;
cout << (d1 != d2) << endl;
return 0;
}
4. 操作符 <
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
bool operator >(const date& d2)// d1>d2
{
if (_year > d2._year)//年大为大
{
return true;
}
else if (_year == d2._year && _month > d2._month) //年相,月大为大
{
return true;
}
else if (_year == d2._year && _month == d2._month && _day > d2._day)//年 月相等,天大为大
{
return true;
}
else
{
return false;
}
}
bool operator<(const date& d2)// d1 < d2
{
return !(*this > d2);
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
d1 < d2;
cout << (d1 < d2) << endl;
return 0;
}
5 .赋值操作符(编译器默认实现)
- 正常使用
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
void operator=(const date& d2)
{
//为了防止自己给自己赋值的事情发生,如:d1=d1
if ( this != &d2)
{
_year = d2._year;
_month = d2._month;
_day = d2._day;
}
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022,12,21);
date d2(2022,12,22);
d1 =d2;
return 0;
}
- 连续赋值情况的考虑
#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 print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
date operator=(const date& d)//传值返回 类型为date
{
//为了防止自己给自己赋值的事情发生,如:d1=d1
if (this != &d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}
return *this;
}
date(const date& d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
date d1(2022, 12, 21);
date d2(d1);
date d3;
d2 = d3 = d1;
int i = 0;
int j = 0;
int z = 0;
i = j = z;
return 0;
}
五. const成员
1.对象调用const成员函数
#include<iostream>
using namespace std;
class date
{
public:
void print()
{
cout << _a << endl;
}
private:
int _a;
};
int main()
{
const date a;
a.print();//权限放大,会报错
return 0;
}
将&a传过去由this指针接收, a指针类型为 const date* ,this指针类型为 date* ,将const date * 传给date * 属于权限的放大。
把this指针类型改成 const date* 就可以了,但是由于this指针是隐藏的,要怎么解决呢?。
#include<iostream>
using namespace std;
class date
{
public:
void print()const //this指针类型变为const date*
{
cout << _a << endl;
}
private:
int _a;
};
int main()
{
const date a;
a.print();//正确 权限保持
return 0;
}
const 在左右对于指针的影响
const date p1 : 修饰的是* p1, const在 * 左面 指针所指向的内容不变
date const * p2 : 修饰的是* p2,const在* 左面 指针所指向的内容不变
date * const p3 : 修饰的是 p3,const在 * 右面,指针本身不变
6.取地址及const取地址操作符重载
六.取地址及const取地址操作符重载
#include<iostream>
using namespace std;
class date
{
public:
date* operator&()
{
return this;
}
const date* operator&()const
{
return this;
}
};
int main()
{
date d;
cout << &d << endl;
return 0;
}