类与对象中
一、类的6个默认成员函数
当类为空类时,会自动生产6个默认成员函数。
二、构造函数
1.概念:构造函数是一个特殊的成员函数,名字与类名相同。在对象的生命周期内只调用一次。
2.特性
(1)函数名与类名相同
(2)无返回值
(3)对象实例化时编译器自动调用对应的构造函数
(4)构造函数可以重载
(5)若类中没有显示定义构造函数,则C++编译器会自动生成一个无参的默认函数
(6)无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个
注:构造函数并不是开空间创建对象,而是初始化对象
class Date
{
public:
//1.无参构造函数
Date()
{}
//2.带参构造函数
Date(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
Date d1; //调用无参构造函数
Date d2(2019, 1, 1); //调用带参构造函数
return 0;
}
(7)C++把类型分成内置类型(基本类型)和自定义类型。编译器生成的默认构造函数会对自定义类型成员调用它的默认成员函数。
内置类型就是语法已经定义好的类型:如int/char…,自定义类型就是我们使用class/struct/union自己定义的类型。
class Time
{
public:
Time()
{
cout << "Time()" << endl;
_hour = 0;
_minute = 0;
_second = 0;
}
private:
int _hour;
int _minute;
int _second;
};
class Date
{
private:
//基本类型(内置类型)
int _year;
int _month;
int _day;
//自定义类型
Time _t;
};
int main()
{
Date d;
return 0;
}
三、析构函数
1、概念:析构函数不是完成对象的销毁,局部对象的销毁工作是由编译器完成的。对象在销毁时会自动调用析构函数,完成一些资源清理工作。
2、特性:
(1)析构函数名是在类名前加上字符~。
(2)无参数无返回值。
(3)一个类有且只有一个析构函数。
(4)对象生命周期结束时,C++编译系统自动调用析构函数。
typedef int Datatype;
class SeqList
{
public:
SeqList(int capacity = 10)
{
_pData = (Datatype*)malloc(sizeof(Datatype)*capacity);
assert(_pData);
_size = 0;
_capacity = capacity;
}
~SeqList()
{
if (_pData)
{
free(_pData);
_pData = NULL;
_size = 0;
_capacity = 0;
}
}
private:
int* _pData;
size_t _size;
size_t _capacity;
};
(5)编译器生成的默认构造函数会对自定义类型成员调用其析构函数,不处理内置类型成员。
class String
{
public:
String(const char* str = "jack")
{
_str = (char*)malloc(strlen(str) + 1);
strcpy(_str, str);
}
~String()
{
cout << "~String()" << endl;
free(_str);
}
private:
char* _str;
};
class Person
{
private:
String _name;
int _age;
};
int main()
{
Person p;
return 0;
}
四、拷贝构造
特征:
(1)拷贝构造函数是构造函数的一个重载形式。
(2)拷贝构造函数的参数只有一个且必须使用引用传参,使用传值方式会引发无穷递归调用。
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(2019, 1, 1);
Date d2(d1);
return 0;
}
若用传值的方式,当d1传参给date时,date要将d1里的内容拷贝下来,这就又需要拷贝构造函数,继续传值后又引发了对象的拷贝,导致了无穷递归调用。而使用引用传参,date时d1的别名,d1传参给date时,不需要进行拷贝,直接使用即可。
(3)若未显示定义,系统生成默认的拷贝构造函数。默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或值拷贝。
class Date
{
public:
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
Date d1(2019, 1, 1);
//这里d2调用的默认拷贝函数,d2的值和d1的值一样
Date d2(d1);
return 0;
}
五、赋值运算符重载
1、运算符重载
运算符重载是具有特殊函数名的函数。
函数名字:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型operator操作符。
注:(1)不能通过连接其他符号来创建新的操作符:比如[email protected]。
(2)重载操作符必须有一个类类型或联合类型的操作数。
(3)用于内置类型的操作符,其含义不能改变。
(4)作为类成员的重载函数时,其形参看起来比操作数数目少1成员函数的。
(操作符有一个默认的形参this,限定为第一个形参)
(5).* 、:: 、sizeof 、? : 、.。注意以上5个运算符不能重载。
class Date
{
public:
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
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(2019, 1, 1);
Date d2(2019, 2, 8);
cout << (d1 == d2) << endl;
return 0;
}
2、赋值运算符重载
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;
}
Date& operator=(const Date& d)
{
if (this != &d)
{
_year = d._year;
_month = d._month;
_day = d._day;
}
return *this;
}
private:
int _year;
int _month;
int _day;
};
六、const成员
1、const修饰类的成员函数
将const修饰的类成员函数称之为const成员函数,实际修饰该成员函数隐含的this指针,表明该成员函数中不能对类的任何成员进行修改。
注:const对象不可以调用非const成员函数,而非const对象可以调用const成员函数;const成员函数内不可以调用其他的非const成员函数,而非const成员函数内可以调用其他的const成员函数
2、取地址及const取地址操作符重载
这两个默认成员函数一般不需要重新定义。
class Date
{
public:
Date* operator&()
{
return this;
}
const Date* operator&()const
{
return this;
}
private:
int _year;
int _month;
int _day;
};