第二周 周赛新知(运算符重载)

    一:重载运算符 

      1.运算符重载定义:

C++中预定义的运算符的操作对象只能是基本数据类型。但实际上,对于许多用户自定义类型(例如类),也需要类似的运算操作。这时就必须在C++中重新定义这些运算符,赋予已有运算符新的功能,使它能够用于特定类型执行特定的操作。运算符重载的实质是函数重载,它提供了C++的可扩展性,也是C++最吸引人的特性之一。

运算符重载是通过创建运算符函数实现的,运算符函数定义了重载的运算符将要进行的操作。运算符函数的定义与其他函数的定义类似,惟一的区别是运算符函数的函数名是由关键字operator和其后要重载的运算符符号构成的。运算符函数定义的一般格式如下:

  <返回类型说明符> operator <运算符符号>(<参数表>)

{

     <函数体>

}

 2.运算符重载时要遵循以下规则:

(1) 除了类属关系运算符"."、成员指针运算符".*"、作用域运算符"::"、sizeof运算符和三目运算符"?:"以外,C++中的所有运算符都可以重载。

(2) 重载运算符限制在C++语言中已有的运算符范围内的允许重载的运算符之中,不能创建新的运算符。

(3) 运算符重载实质上是函数重载,因此编译程序对运算符重载的选择,遵循函数重载的选择原则。

(4) 重载之后的运算符不能改变运算符的优先级和结合性,也不能改变运算符操作数的个数及语法结构。

(5) 运算符重载不能改变该运算符用于内部类型对象的含义。它只能和用户自定义类型的对象一起使用,或者用于用户自定义类型的对象和内部类型的对象混合使用时。

(6) 运算符重载是针对新类型数据的实际需要对原有运算符进行的适当的改造,重载的功能应当与原有功能相类似,避免没有目的地使用重载运算符。

(7)重载运算符的函数不能有默认的参数,否则就改变了运算符的参数个数,与前面第3点相矛盾了;

(8)重载的运算符只能是用户自定义类型,否则就不是重载而是改变了现有的C++标准数据类型的运算符的规则了,会引会天下大乱的;

(9)用户自定义类的运算符一般都必须重载后方可使用,但两个例外,运算符=&不必用户重载;

(10)运算符重载可以通过成员函数的形式,也可是通过友元函数,非成员非友元的普通函数。

 

3.运算符重载的形式:

运算符函数重载一般有两种形式:重载为类的成员函数和重载为类的非成员函数。非成员函数通常是友元。(可以把一个运算符作为一个非成员、非友元函数重载。但是,这样的运算符函数访问类的私有和保护成员时,必须使用类的公有接口中提供的设置数据和读取数据的函数,调用这些函数时会降低性能。可以内联这些函数以提高性能。)   

1) 成员函数运算符

 运算符重载为类的成员函数的一般格式为:

    <函数类型> operator <运算符>(<参数表>)

    {

     <函数体>

    }

 当运算符重载为类的成员函数时,函数的参数个数比原来的操作数要少一个(后置单目运算符除外),这是因为成员函数用this指针隐式地访问了类的一个对象,它充当了运算符函数最左边的操作数。因此:

(1) 双目运算符重载为类的成员函数时,函数只显式说明一个参数,该形参是运算符的右操作数。

(2) 前置单目运算符重载为类的成员函数时,不需要显式说明参数,即函数没有形参。

(3) 后置单目运算符重载为类的成员函数时,函数要带有一个整型形参。

    调用成员函数运算符的格式如下:

    <对象名>.operator <运算符>(<参数>)

    它等价于

    <对象名><运算符><参数>

 例如:a+b等价于a.operator +(b)。一般情况下,我们采用运算符的习惯表达方式。

2) 友元函数运算符

 运算符重载为类的友元函数的一般格式为:

    friend <函数类型> operator <运算符>(<参数表>)

   {

     <函数体>

    }

当运算符重载为类的友元函数时,由于没有隐含的this指针,因此操作数的个数没有变化,所有的操作数都必须通过函数的形参进行传递,函数的参数与操作数自左至右一一对应。

 调用友元函数运算符的格式如下:

    operator <运算符>(<参数1>,<参数2>)

    它等价于

    <参数1><运算符><参数2>

    例如:a+b等价于operator +(a,b)

4.两种重载形式的比较

在多数情况下,将运算符重载为类的成员函数和类的友元函数都是可以的。但成员函数运算符与友元函数运算符也具有各自的一些特点:

(1) 一般情况下,单目运算符最好重载为类的成员函数;双目运算符则最好重载为类的友元函数。

(2) 以下一些双目运算符不能重载为类的友元函数:=、()、[]、->。

(3) 类型转换函数只能定义为一个类的成员函数而不能定义为类的友元函数。

(4) 若一个运算符的操作需要修改对象的状态,选择重载为成员函数较好。

(5) 若运算符所需的操作数(尤其是第一个操作数)希望有隐式类型转换,则只能选用友元函数。

(6) 当运算符函数是一个成员函数时,最左边的操作数(或者只有最左边的操作数)必须是运算符类的一个类对象(或者是对该类对象的引用)。如果左边的操作数必须是一个不同类的对象,或者是一个内部类型的对象,该运算符函数必须作为一个友元函数来实现。

(7) 当需要重载运算符具有可交换性时,选择重载为友元函数。

5.实例:

 1)用成员函数来重载运算符:

#include <iostream>

using namespace std;

class X{

   int i;

public:

   X(int ii=0){i=ii;}

   X operator   +(const X &rx){

       i+=rx.i;

       return X(i);

   }

   int GetI(){return i;}

};

int main(){

   X a(1),b(3);

   cout<<(a+b).GetI()<<endl;

   return 0;

}

 

2) 用友元函数来重载运算符

说明:此时若用

#include <iostream>

using namespace std;

则会出现如下的错误:

fatal error C1001: INTERNAL COMPILER ERROR

所以头文件用:#include<iostream.h>

 

#include <iostream>

using namespace std;

class Complex{

public:

   Complex(double r=0.0,double i=0.0){

       real=r;

       image=i;

   }

   friend Complex operator+(const Complex&,const Complex&);

   void display();

private:

   double real;

   double image;

};

 

Complex operator+(const Complex &c1,const Complex &c2){

   return Complex(c1.real+c2.real,c1.image+c2.image);

}

 

void Complex::display(){

   cout<<"("<<real<<","<<image<<"i)"<<endl;

}

 

int main(){

   Complex c1(3,4),c2(5,-10),c3;

   c3=c1+c2;

   cout<<"c1=";c1.display();

   cout<<"c2=";c2.display();

   cout<<"c1+c2=";c3.display();

   return 0;

}

—————————————————————————————————————————————————————————————————————————————

二:getline函数

          学习C++的同学可能都会遇到一个getline()函数,譬如在C++premer中,标准string类型第二小节就是“用getline读取整行文本”。书上给的程序如下:
 int main()
{
    string line:
   while(getline(cin,line))
   cout<<line<<endl;
    return 0;
}
大家会发现运行时怎么也跳不出循环,甚至会发生各种莫名其妙的错误。这是为什么呢?在这里我给大家做一个详细的讲解。
首先给大家介绍一下getline()函数(个人觉得百度百科给的果断不够详细)
大家百度会发现getline()的原型是istream&getline ( istream &is , string &str, char delim );
其中 istream &is表示一个输入流,譬如cin;string&str表示把从输入流读入的字符串存放在这个字符串中(可以自己随便命名,str什么的都可以);chardelim表示遇到这个字符停止读入,在不设置的情况下系统默认该字符为'\n',也就是回车换行符(遇到回车停止读入)。给大家举个例子:
string line;
cout<<"please cin a line:"
getline(cin,line,'#');
cout<<endl<<"Theline you give is:"line;
那么当我输入"You are the #best!"的时候,输入流实际上只读入了"You are the",#后面的并没有存放到line中(应该是在缓冲区里吧)。然后程序运行结果应该是这样的:
 please cin aline:You are the #best!
 The line you giveis:You are the 
而且这里把终止符设为#,你输入的时候就算输入几个回车换行也没关系,输入流照样会读入,譬如:
                         please cin a line:You are thebest!
//这里输入了一个回车换行
     Thank you!
      //终止读入
     The line yougive is:You are the best!
//换行照样读入并且输出
       Thankyou!
以上就是getline()函数一个小小的实例了。
那么如果把getline()作为while的判断语句会怎么样呢?
让我们一起来分析一下while(getline(cin,line))语句
注意这里默认回车符停止读入,按Ctrl+Z或键入EOF回车即可退出循环。
在这个语句中,首先getline从标准输入设备上读入字符,然后返回给输入流cin,注意了,是cin,所以while判断语句的真实判断对象是cin,也就是判断当前是否存在有效的输入流。在这种情况下,我想只要你的电脑不中毒不发神经你的输入流怎么会没有效?所以这种情况下不管你怎么输入都跳不出循环,因为你的输入流有效,跳不出循环。
然而有些同学误以为while判断语句的判断对象是line(也就是line是否为空),然后想通过直接回车(即输入一个空的line)跳出循环,却发现怎么也跳不出循环。这是因为你的回车只会终止getline()函数的读入操作。getline()函数终止后又进行while()判断(即判断输入流是否有效,你的输入流当然有效,满足条件),所以又运行getline()函数,所以,你懂了吧。。。

     三:显隐示转换

        (1)  定义: 一般情况下,数据的类型的转换通常是由编译系统自动进行的,不需要人工干预,所以被称为隐式类型转换。但如果程序要求一定要将某一类型的数据转换为另外一种类型,则可以利用强制类型转换运算符进行转换,这种强制转换过程称为显式转换。

        (2)  格式:   (强制的类型名)表达式      用于强行将“表达式”的值转换成“类型名”所表示的数据类型。例如,(int)4.2的结果是4;强制类型转换的目的是使表达式的值的数据类型发生改变,从而使不同类型数据之间的运算能够进行下去。

        (3)  应用情况:   如果表达式仅是单个常量或变量,则常量或变量不必用圆括号括起来;但是如果是含有运算符的表达式,则必须利用括号将其统一,否则容易发生歧义。例如:
(float)(a+b),(int)a+b ;经强制类型转换后仅产生一个临时的、类型不同的数据继续参加运算,其常量、变量或表达式的原有类型以及原来数据值均不改变。例如:
 int x,(float)x; 此时x的值被强制转换为实型参与下一步的运算处理,但并不改变 x 的类型是整型。
这里举个形象的例子,比如说你是个男生,但是你男扮女装(相当于强制转换了类型),和阿姨妩媚羡慕的打了个招呼,然后混进了女生宿舍,什么!!! 门口阿姨竟然没看出来!!!。然后你就可以像其她女生一样在宿舍楼里活动了,但是!———————但是你现在还是个男生。 除非大笑(这里就相当于赋值--------------x = (float) X )
           注意:   由于类型转换将占用系统时间,过多的转换将降低程序的运行效率。再设计程序时应尽量选择好数据类型,以减少不必要的类型转换。


posted @ 2014-08-09 21:13  6bing  阅读(183)  评论(0编辑  收藏  举报