C++第11章,使用类

来源:互联网 发布:夏普百视通网络电视 编辑:程序博客网 时间:2024/06/12 19:21

运算符重载
operator+(argument-list)
假设有一个Saleperson类
如果a,b,c都是其对象。
a = b + c;
等价于
a = b.operator+(c)隐式的使用b,显式的使用c

a = b + c + d ;
a = b.operator+( c + d)
a = b.operator+(c.operator+(d));
重载限制:
重载后的运算符的操作数必须有一个是用户自定义的类型,这样可以防止用户为标准类型的操作重载运算符。比如不能把-重载为两个double类型的和。而不是它们的差。

11.3友元

友元函数

友元类

友元成员函数

通过让函数成为类的友元,可以赋予该函数与类的成员函相同的访问权限。

非成员函数不是由对象调用的,它使用的所有值(包括对象)都是显式参数。

A = 2.75 * B;
与下面的非成员函数调用匹配:
A = operator*(2.75,B);
该函数的原型如下:
Time operator*(double m,const Time & t);

有一类特殊的非成员函数可以访问类的私有数据,他们被称为友元函数。

C++编程常见错误—cannot have cv-qualifier//不能有CV限定,在C++中CV指const和volatile—1、非成员函数不能有CV限定,2、静态成员函数不能有CV限定

提示:如果要为类重载运算符,并将非类的项作为其第一个操作数,则可以用友元函数来翻转操作数。

#include <iostream>using namespace std;class Time{private:    int hours;    int minutes;public:    Time();    Time(int h,int m = 0);    void addMin(int m);    void addHr(int h);    void Reset(int h = 0,int m = 0);    Time operator+(const Time & t) const;    Time operator*(int mult)const;    friend Time operator*(int m,const Time & t);    void show()const;};Time::Time(){    hours = minutes = 0;}Time::Time(int h,int m){    hours = h;    minutes = m;}void Time::addMin(int m){    minutes += m;    hours += minutes / 60;    minutes %= 60;}void Time::addHr(int h){    hours += h;}void Time::Reset(int h ,int m){    hours = h;    minutes = m;}Time Time::operator+(const Time & t)const{    Time sum;    sum.minutes = minutes + t.minutes;    sum.hours = hours + t.hours + sum.minutes / 60;    sum.minutes %= 60;    return sum;}Time Time::operator*(int mult)const{    Time result;    result.hours = hours + mult*minutes / 60 ;    result.minutes = mult*minutes % 60 ;    return result;}void Time::show()const{    std::cout << hours << "hours," << minutes << "minutes";}Time operator*(int m,const Time & t){    Time result;    result.minutes = t.minutes * m % 60;    result.hours = t.hours + t.minutes * m / 60;    return result;}int main(){    /*Time planning;    Time coding(2,40);    Time fixing(5,55);    Time total;    cout << "planning time = ";    planning.show();    cout << endl;    cout << "coding time = ";    coding.show();    cout << endl;    cout << "fixing time = ";    fixing.show();    cout << endl;    total = coding + fixing;    cout << "coding.Sum(fixing) = ";    total.show();    cout << endl;    return 0;*/    Time a(1,35);    Time b;    b = 2 * a;    b.show();}

上述对定义进行修改,也可以不使用友元函数,可以将这个友元函数编写为非友元函数

Time operator*(int m,const Time & t){    return t * m;//use t.operator*(m);}

原来的版本显式的访问类的私有成员,所以必须是友元函数,这个版本让成员函数来处理私有值,所以不必是友元函数。

常用的友元: 重载<<运算符
要使Time类知道使用cout,必须使用友元函数。如果使用一个Time成员函数来重载<<,Time对象将是第一个操作数。这意味着必须这样使用<<
trip << cout
但是通过使用友元函数,可以像下面这样重载运算符:
void operator<<(ostream & os,const Time &t){
os << t.hours << “hours ,” << t.minutes << “minutes”;
}

但是返回类型为void,意味着无法cout << x << y 这样使用。
为此可更改为:
ostream& operator<<(ostream & os,const Time &t){
os << t.hours << “hours ,” << t.minutes << “minutes”;
return os;
}

代码很短时,应该使用内联函数。inline。一般将定义和声明放在一起。如果分开,则只在声明中使用inline即可。

11.6 类的自动转换和强制类型转换。
类型转换构造函数:
在C++中,接受一个参数的构造函数,为将类型和该参数相同的值转换为类对象提供了蓝图。
下面的构造函数用于将double类型的值转换为Stonewt类型
Stonewt(double lbs)
Stonewt myCat;
myCat = 19.6;
程序将使用19.6创造一个临时对象并且初始化。随后将赋值给myCat;这一过程称之为隐式转换,因为它是自动进行的。
只有接受一个参数的构造函数才能作为转换函数。但是如果出了第一个参数,其余形参提供了默认值,则可以用于转换。
比如
Stonewt(int a,int b = 0);则可以用于转换int
C++提供了关键字explicit用于关闭这项特性。
explicit Stonewt(int a);
仅仅关闭了隐式类型转换,但是仍然可以进行强制类型转换。

转换函数:
以上将数字转换为Stoewt对象。可以做相反的转换吗?

0 0
原创粉丝点击