赞
踩
我们前面学习了关于类和对象的6个默认成员函数,知道了一个空类中有构造函数和析构函数,通过对对象初始化和对象中进行资源清理,达到初始化和销毁的作用。我们再对一些小的点进行补充,看看类和对象的一些细节。
class Date
{
public:
Date(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
class Date
{
public:
Date(int year, int month, int day)
: _year(year)
, _month(month)
, _day(day)
{}
private:
int _year;
int _month;
int _day;
};
【注意】
class A { public: A(int a) :_a(a) {} private: int _a; }; class B { public: B(int a, int ref) :_aobj(a) // 进行显示调用 , _ref(ref) , _n(10) {} private: A _aobj; // 没有默认构造函数 int& _ref; // 引用 const int _n; // const };
class Time { public: Time(int hour = 0) :_hour(hour) { cout << "Time()" << endl; } private: int _hour; }; class Date { public: Date(int day) {} private: int _day; Time _t; }; int main() { Date d(1); }
class A { public: A(int a) :_a1(a) , _a2(_a1) {} void Print() { cout << _a1 << " " << _a2 << endl; } private: int _a2; int _a1; }; int main() { A aa(1); aa.Print(); }
A.输出1 1
B.程序崩溃
C.编译不通过
D.输出1 随机值
D
,初始化列表初始化的是声明的顺序构造函数不仅可以构造与初始化对象,对于单个参数或者除第一个参数无默认值其余均有默认值的构造函数,还具有类型转换的作用。
类型转换,在我们之前c语言中学到,一个int类型赋值给一个double类型,会发生类型转换,一个是8个字节,一个是4个字节(32位),在一定的精度丢失(四舍五入)后就会完成隐式的类型转换。
在对象中,举个例子:Date a1 = 1;这里的1是先构造了一个无名对象,然后把这个无名对象的值赋值给了我们的a1,这里是一个隐式的类型转换。
class Date { public: // 1. 单参构造函数,没有使用explicit修饰,具有类型转换作用 // explicit修饰构造函数,禁止类型转换---explicit去掉之后,代码可以通过编译 explicit Date(int year) :_year(year) {} /* // 2. 虽然有多个参数,但是创建对象时后两个参数可以不传递,没有使用explicit修饰,具 有类型转换作用 // explicit修饰构造函数,禁止类型转换 explicit Date(int year, int month = 1, int day = 1) : _year(year) , _month(month) , _day(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; }; void Test() { Date d1(2022); // 用一个整形变量给日期类型对象赋值 // 实际编译器背后会用2023构造一个无名对象,最后用无名对象给d1对象进行赋值 d1 = 2023; // 将1屏蔽掉,2放开时则编译失败,因为explicit修饰构造函数,禁止了单参构造函数类型转换的作用 }
上述代码可读性不是很好,用explicit修饰构造函数,将会禁止构造函数的隐式转换。
explicit为清晰的;明确的之意.顾名思义,关键字explicit可以阻止隐式转换的发生.
例如: C++中只带有一个参数的构造函数,或者或者除了第一个参数外其余参数都有缺省值的多参构造函数,承担了两个角色:
1.用于构建单参数的类对象.
2.隐含的类型转换操作符.
例如:一个类A的构造函数A(int i)就是,既可以用来作为构造器,又可以实现隐式转换A a=1;因为1可以通过构造函数A(int i)转换为一个类A的对象。(隐含的类型转换操作符)
但有时我们并不想让他进行隐式类型转换,这时C++的explicit关键字就起到作用了.
面试题:实现一个类,计算程序中创建出了多少个类对象。
class A { public: A() { ++_scount; } A(const A& t) { ++_scount; } ~A() { --_scount; } static int GetACount() { return _scount; } private: static int _scount; }; // 放在外面进行初始化 int A::_scount = 0; void main() { cout << A::GetACount() << endl; A a1, a2; A a3(a1); cout << A::GetACount() << endl; }
【问题】
友元提供了一种突破封装的方式,有时提供了便利。但是友元会增加耦合度,破坏了封装,所以友元不宜多用。
友元分为:友元函数和友元类
class Date { public: Date(int year, int month, int day) : _year(year) , _month(month) , _day(day) {} // d1 << cout; -> d1.operator<<(&d1, cout); 不符合常规调用 // 因为成员函数第一个参数一定是隐藏的this,所以d1必须放在<<的左侧 ostream& operator<<(ostream& _cout) { _cout << _year << "-" << _month << "-" << _day << endl; return _cout; } private: int _year; int _month; int _day; };
class Date { friend ostream& operator<<(ostream& _cout, const Date& d); friend istream& operator>>(istream& _cin, Date& d); public: Date(int year = 1900, int month = 1, int day = 1) : _year(year) , _month(month) , _day(day) {} private: int _year; int _month; int _day; }; ostream& operator<<(ostream& _cout, const Date& d) { _cout << d._year << "-" << d._month << "-" << d._day; return _cout; } istream& operator>>(istream& _cin, Date& d) { _cin >> d._year; _cin >> d._month; _cin >> d._day; return _cin; } int main() { Date d; cin >> d; cout << d << endl; return 0; }
说明:
特性:
class A { private: static int k; int h; public: class B // B天生就是A的友元 { public: void foo(const A& a) { cout << k << endl;//OK cout << a.h << endl;//OK } }; }; int A::k = 1; int main() { A::B b; b.foo(A()); return 0; }
class A { public: A(int a = 0) :_a(a) { cout << "A(int a)" << endl; } ~A() { cout << "~A()" << endl; } private: int _a; }; class Solution { public: int Sum_Solution(int n) { //... return n; } }; int main() { A aa1; // 不能这么定义对象,因为编译器无法识别下面是一个函数声明,还是对象定义 //A aa1(); // 但是我们可以这么定义匿名对象,匿名对象的特点不用取名字, // 但是他的生命周期只有这一行,我们可以看到下一行他就会自动调用析构函数 A(); A aa2(2); // 匿名对象在这样场景下就很好用,当然还有一些其他使用场景,这个我们以后遇到了再说 Solution().Sum_Solution(10); return 0; }
class A { public: A(int a = 0) :_a(a) { cout << "A(int a)" << endl; } A(const A& aa) :_a(aa._a) { cout << "A(const A& aa)" << endl; } A& operator=(const A& aa) { cout << "A& operator=(const A& aa)" << endl; if (this != &aa) { _a = aa._a; } return *this; } ~A() { cout << "~A()" << endl; } private: int _a; }; void f1(A aa) {} A f2() { A aa; return aa; } int main() { // 传值传参 A aa1; f1(aa1); cout << endl; // 传值返回 f2(); cout << endl; // 隐式类型,连续构造+拷贝构造->优化为直接构造 f1(1); // 一个表达式中,连续构造+拷贝构造->优化为一个构造 f1(A(2)); cout << endl; // 一个表达式中,连续拷贝构造+拷贝构造->优化一个拷贝构造 A aa2 = f2(); cout << endl; // 一个表达式中,连续拷贝构造+赋值重载->无法优化 aa1 = f2(); cout << endl; return 0; }
现实生活中的实体计算机并不认识,计算机只认识二进制格式的数据。如果想要让计算机认识现实生活中的实体,用户必须通过某种面向对象的语言,对实体进行描述,然后通过编写程序,创建对象后计算机才可以认识。比如想要让计算机认识洗衣机,就需要:
class Solution { class Sum { public: Sum() { _ret += _i; _i++; } static int GetRet() { return _ret; } }; public: // sum是Solution的友元 int Sum_Solution(int n) { Sum a[n]; return _ret; } private: static int _i; static int _ret; }; int Solution::_i = 1; int Solution::_ret = 0;
#include <iostream> using namespace std; int main() { int year, mon, day; int data[13] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; while (cin >> year >> mon >> day) { int sum = 0; for (int i = 1; i <= mon; i++) { sum += data[i]; } if ((mon == 2) && ((year % 4 == 0) && (year % 100 != 0) || (year % 400 == 0))) { // 如果是闰年就要多加1天 sum = sum + 1 + day; } else { // 需要加上最后的天数 sum = sum +day; } cout << sum << endl; } return 0; }
思路:
//平年从1月到n月的天数 static int mon[12] = {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334}; //给出年月日,计算距离0000年0月1日的天数和 int CountDay(int y, int m, int d) { // 计算0-y年的天数 int yearDay = y * 365 + y / 4 - y / 100 + y / 400; // 计算到0-m月的天数 int monthDay = mon[m - 1]; if (m > 2 && ((y % 4 == 0 && y % 100 != 0) || y % 400 == 0)) monthDay += 1; return yearDay + monthDay + d; } int main() { int year1, month1, day1; scanf("%4d%2d%2d", &year1, &month1, &day1); int n1 = CountDay(year1, month1, day1); int year2, month2, day2; scanf("%4d%2d%2d", &year2, &month2, &day2); int n2 = CountDay(year2, month2, day2); cout << abs(n1 - n2) + 1 << endl; }
#include <iostream> using namespace std; int main() { int year; int day; int mon[13] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; while (cin >> year >> day) { if ((year % 4 == 0) && (year % 100 != 0) || (year % 400 == 0)) mon[2] = 29; else mon[2] = 28; for (int i = 1; i <= 12; i++) { if (day <= mon[i]) { printf("%04d-%02d-%02d\n", year, i, day); break; } else { day -= mon[i]; } } } }
#include <iostream> using namespace std; int main() { int m = 0; while (cin >> m) { int year, mon, day, num; for (int i = 0; i < m; i++) { static int days[13] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; cin >> year >> mon >> day >> num; day += num; while (day > days[mon]) { if ((year % 4 == 0) && (year % 100 != 0) || (year % 400 == 0)) days[2] = 29; else days[2] = 28; day -= days[mon]; if (day == 0) { day = 1; } mon++; if (mon == 13) { year++; mon = 1; } } printf("%4d-%02d-%02d\n", year, mon, day); } } }
最后本文就到这里结束了,感谢大家的收看,请多多指点~
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。