什么是构造函数
类通过一个或者多个特殊的成员函数来控制其对象的初始化过程,其任务是初始化类对象的数据成员。
构造函数的一些特性:
1. 无论何时只要类的对象被创建,编译器就自动调用构造函数,而且在对象的生命周期内只且只调用一次。
2. 构造函数名字与类名相同,且没有返回值。
3. 构造函数可以重载,不同的构造函数之间必须在参数数量或者类型上有所区别
4. 构造函数不能用const来修饰,因为当我们创建类的一个const对象,直到构造函数完成初始化过程,对象才真正取得其“常量”属性,因此构造函数在const对象的过程中可以向其写值。
class Date {
public:
Date(){}
Date(intyear , int month, day) :_year(year), _month(month), _day(day) {}
private:
int_year;
int_month;
int_day;
};
上面就是一个Date类,有一个默认的无参构造函数Date(),一个有参的构造函数Date(int year, int month, day),形成重载。需要注意的是,当类中没有显示定义构造函数,那么编译器就会为我们隐式定义一个默认构造函数形如Date(),这个默认构造函数没有参数,又被称为合成的默认构造函数。
另外无参构造函数和带有缺省值的构造函数都认为是缺省构造函数,并且缺省构造函数只能有一个。
Date();
Date(int year = 0, int month = 0, int day =0) :_year(year), _month(month), _day(day) {}
这样写是不能编译通过的。
构造函数初始化列表:以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个数据成员后面跟一个放在园括号中的初始化式形如Date(int year , int month, day) :_year(year), _month(month),_day(day) {}
在一些场合,比如数据成员为内置类型时,初始化列表相当于赋值
_year = year;
_month = month;
_day = day;
然而类中包含以下成员必须要放在初始化列表中初始化:
1、引用数据成员 2、const数据成员 3、类类型成员(该类没有缺省的构造函数)
拷贝构造函数
只有单个形参,而且该形参是对本类类型对象的引用(常用const修饰),这样的构造函数称为拷贝构造函数。拷贝构造函数是特殊的构造函数,创建对象时使用已存在的同类对象来进行初始化,由编译器自动调用。
Date( const Date& date)
{ _year = date._year; _month = date.month;_day = date._day; }
类型转换构造函数,根据一个指定的类型的对象创建一个本类的对象.
比如定义一个Date(int year) :_year(year) {}
在调用时候
Date d = 2016;
对于出现这种单参数的构造函数,C++会默认将参数对应的类型转换为该类类型,有时候这种隐私的转换是我们所不想要的,所以需要使用explicit来限制这种转换。
this指针
当我们在进入一个房子之后,可以看见房子里的东西,但是看不到房子的全貌。对于一个类的实例来说,你可以看到它的成员函数、成员变量,但是实例本身呢?this是一个指针,它时时刻刻指向这个实例。
特性:
1、this指针的类型 类类型* const。
2、this指针并不是对象本身的一部分,不影响sizeof的结果。
3、this的作用域在类成员函数的内部。
4、this指针是类成员函数的第一个默认隐含参数,编译器自动维护传递,类编写者不能显式传递。
class A
{
public:
constint get(){return i;}
voidset(int x){this->i=x;cout<<"this指针保存的内存地址为:"<<this<<endl;}
private:
int i;
};
int main()
{
Aa;
a.set(1);
cout<<"对象a所在的内存地址为:"<<&a<<endl;
cout<<"对象a所保存的值为:"<<a.get()<<endl;
cout<<endl;
Ab;
b.set(11);
cout<<"对象b所在的内存地址为:"<<&b<<endl;
cout<<"对象b所保存的值为:"<<b.get()<<endl;
return 0;
}