1.为什么要有构造函数?
class Date
{
public:
void Init(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
void Print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
Date d1;
d1.Init(2022, 7, 5);
d1.Print();
Date d2;
d2.Init(2022, 7, 6);
d2.Print();
return 0;
}
对于Date类,可以通过 Init 公有方法给对象设置日期,但如果每次创建对象时都调用该方法设置信息,未免有点麻烦,那能否在对象创建时,就将信息设置进去呢?答案:使用默认构造函数~
class Date
{
public:
Date(int year = 10,int month = 1,int day = 2)
{
_year = year;
_month = month;
_day = day;
}
void Print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month ;
int _day ;
};
int main()
{
Date d1;
cout << sizeof(Date) << endl;//构造函数的目的就是为了对对象进行初始化 不是占位 也不是为了开辟空间 就是为了初始化
cout << sizeof(d1) << endl;
return 0;
}
由此可见,默认构造函数存在的目的就是规范类对象的实例化,使其初始化更加方便,既不是占位,也不是更好的开辟空间,就是为了初始化!
2.构造函数的特征
形式:没有返回值 函数名必须与类名名称相同 自动调用 可以重载
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;
};
void TestDate()
{
Date d1; // 调用无参构造函数
Date d2(2015, 1, 1); // 调用带参的构造函数
// 注意:如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明
//编译器不知道他是无参的构造函数还是函数声明 容易产生歧义
Date d3();
}
请看报错信息:
由此可知,如果通过无参的构造函数创建对象,就不用写后边的括号,否则会和函数声明的形式相同,产生歧义。
class Date
{
public:
/*
// 如果用户显式定义了构造函数,编译器将不再生成
Date(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
*/
void Print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
// 将Date类中构造函数屏蔽后,代码可以通过编译,因为编译器生成了一个无参的默认构造函数
// 将Date类中构造函数放开,代码编译失败,因为一旦显式定义任何构造函数,编译器将不再生成
// 无参构造函数,放开后报错:error C2512: “Date”: 没有合适的默认构造函数可用
Date d1;
return 0;
}
如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。
3.推荐写全缺省构造函数
//默认构造函数对内置类型(int,char等类型)的成员变量没有操作 但是会调用自定义类型的成员变量默认构造函数
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;
}
由此可见,默认构造函数对内置类型(int,char等类型)的成员变量没有操作 但是会调用自定义类型的成员变量的默认构造函数,所以我们如果不想写默认构造函数,可以在成员变量声明时就给出默认值进行内置类型成员变量的初始化。
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 = 1970;
int _month = 1;
int _day = 1;
// 自定义类型
Time _t;
};
int main()
{
Date d;
return 0;
}
class Date
{
public:
Date()
{
_year = 1900;
_month = 1;
_day = 1;
}
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
// 以下测试函数能通过编译吗?
void Test()
{
Date d1;
//Date d2(1, 2, 3);
}
但是如果不进行传参,既写无参的构造函数和全缺省的构造函数容易出现歧义,编译器不知道调用哪个进行对象初始化而报错,并且为了初始化的更加方便,我们建议就写一个全缺省的默认构造函数。(默认构造函数包括:全缺省构造函数,无参的构造函数,编译器自动生成的构造函数)