• C++iostream以及左右移运算符重载2

    普通类
    • 支持
    • 批判
    • 提问
    • 解释
    • 补充
    • 删除
    • C++iostream以及左右移运算符重载2

     接下来我们继续看一下C++风格的串流控制,C++引入了ostringstream、istringstream、stringstream这三个类,要使用他们创建对象就必须包含sstream.h头文件。
      istringstream类用于执行C++风格的串流的输入操作。
      ostringstream类用于执行C++风格的串流的输出操作。
      stringstream类同时可以支持C++风格的串流的输入输出操作。

      istringstream类是从istream(输入流类)和stringstreambase(c++字符串流基类)派生而来,ostringstream是从ostream(输出流类)和stringstreambase(c++字符串流基类)派生而来,stringstream则是从iostream(输入输出流类)和和stringstreambase(c++字符串流基类)派生而来。

    他们的继承关系如下图所示:

    



     

      istringstream是由一个string对象构造而来,istringstream类从一个string对象读取字符。
      istringstream的构造函数原形如下:
      istringstream::istringstream(string str);

    示例代码如下:
     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <sstream>
    using namespace std;
    int main()
    {
    istringstream istr;
    istr.str("1 56.7",);
    //上述两个过程可以简单写成 istringstream istr("1 56.7");
    cout << istr.str()<<endl;
    int a;
    float b;
    istr>>a;
    cout<<a<<endl;
    istr>>b;
    cout<<b<<endl;
    system("pause");
    }




      上例中,构造字符串流的时候,空格会成为字符串参数的内部分界,例子中对a,b对象的输入"赋值"操作证明了这一点,字符串的空格成为了整型数据与浮点型数据的分解点,利用分界获取的方法我们事实上完成了字符串到整型对象与浮点型对象的拆分转换过程。
      str()成员函数的使用可以让istringstream对象返回一个string字符串(例如本例中的输出操作(cout<<istr.str();)。

    



        ostringstream同样是由一个string对象构造而来,ostringstream类向一个string插入字符。
      ostringstream的构造函数原形如下:
      ostringstream::ostringstream(string str);

    示例代码如下:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <sstream>
    #include <string>
    using namespace std;
    int main()
    {
    ostringstream ostr;
    //ostr.str("abc");//如果构造的时候设置了字符串参数,那么增长操作的时候不会从结尾开始增加,而是修改原有数据,超出的部分增长
    ostr.put('d');
    ostr.put('e');
    ostr<<"fg";

    string gstr = ostr.str();
    cout<<gstr;
    system("pause");
    }




      在上例代码中,我们通过put()或者左移操作符可以不断向ostr插入单个字符或者是字符串,通过str()函数返回增长过后的完整字符串数据,但值得注意的一点是,当构造的时候对象内已经存在字符串数据的时候,那么增长操作的时候不会从结尾开始增加,而是修改原有数据,超出的部分增长。

    



        对于stringstream了来说,不用我多说,大家也已经知道它是用于C++风格的字符串的输入输出的。
      stringstream的构造函数原形如下:

      stringstream::stringstream(string str);

     C++ 代码 示例代码如下:

    //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <sstream>
    #include <string>
    using namespace std;

    int main()
    {
    stringstream ostr("ccc");
    ostr.put('d');
    ostr.put('e');
    ostr<<"fg";
    string gstr = ostr.str();
    cout<<gstr<<endl;

    char a;
    ostr>>a;
    cout<<a

    system("pause");
    }




      除此而外,stringstream类的对象我们还常用它进行string与各种内置类型数据之间的转换。

    示例代码如下:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <sstream>
    #include <string>
    using namespace std;

    int main()
    {
    stringstream sstr;
    //--------int转string-----------
    int a=100;
    string str;
    sstr<<a;
    sstr>>str;
    cout<<str<<endl;
    //--------string转char[]--------
    sstr.clear();//如果你想通过使用同一stringstream对象实现多种类型的转换,请注意在每一次转换之后都必须调用clear()成员函数。
    string name = "colinguan";
    char cname[200];
    sstr<<name;
    sstr>>cname;
    cout<<cname;
    system("pause");
    }




    



        接下来我们来学习一下输入/输出的状态标志的相关知识,C++中负责的输入/输出的系统包括了关于每一个输入/输出操作的结果的记录信息。这些当前的状态信息被包含在io_state类型的对象中。io_state是一个枚举类型(就像open_mode一样),以下便是它包含的值。

    goodbit 无错误

    Eofbit 已到达文件尾

    failbit 非致命的输入/输出错误,可挽回

    badbit 致命的输入/输出错误,无法挽回


      有两种方法可以获得输入/输出的状态信息。一种方法是通过调用rdstate()函数,它将返回当前状态的错误标记。例如,假如没有任何错误,则rdstate()会返回goodbit.

    下例示例,表示出了rdstate()的用法:

     C++ 代码
    //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    int main()
    {
    int a;
    cin>>a;
    cout<<cin.rdstate()<<endl;
    if(cin.rdstate() == ios::goodbit)
    {
    cout<<"输入数据的类型正确,无错误!"<<endl;
    }
    if(cin.rdstate() == ios_base::failbit)
    {
    cout<<"输入数据类型错误,非致命错误,可清除输入缓冲区挽回!"<<endl;
    }
    system("pause");
    }





    另一种方法则是使用下面任何一个函数来检测相应的输入/输出状态:

    bool bad();

    bool eof();

    bool fail();

    bool good();

    下例示例,表示出了上面各成员函数的用法:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    int main()
    {
    int a;
    cin>>a;
    cout<<cin.rdstate()<<endl;
    if(cin.good())
    {
    cout<<"输入数据的类型正确,无错误!"<<endl;
    }
    if(cin.fail())
    {
    cout<<"输入数据类型错误,非致命错误,可清除输入缓冲区挽回!"<<endl;
    }
    system("pause");
    }




    



        如果错误发生,那么流状态既被标记为错误,你必须清除这些错误状态,以使你的程序能正确适当地继续运行。要清除错误状态,需使用clear()函数。此函数带一个参数,它是你将要设为当前状态的标志值。,只要将ios::goodbit作为实参。

    示例代码如下:

     C++ 代码
    //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    int main()
    {
    int a;
    cin>>a;
    cout<<cin.rdstate()<<endl;
    cin.clear(ios::goodbit);
    cout<<cin.rdstate()<<endl;
    system("pause");
    }





      通常当我们发现输入有错又需要改正的时候,使用clear()更改标记为正确后,同时也需要使用get()成员函数清除输入缓冲区,以达到重复输入的目的。

    示例代码如下:
     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    using namespace std;

    int main()
    {
    int a;
    while(1)
    {
    cin>>a;
    if(!cin)//条件可改写为cin.fail()
    {
    cout<<"输入有错!请重新输入"<<endl;
    cin.clear();
    cin.get();
    }
    else
    {
    cout<<a;
    break;
    }
    }
    system("pause");
    }




    最后再给出一个对文件流错误标记处理的例子,巩固学习,代码如下:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <fstream>
    using namespace std;

    int main()
    {
    ifstream myfile("c:\\1.txt",ios_base::in,0);
    if(myfile.fail())
    {
    cout<<"文件读取失败或指定文件不存在!"<<endl;
    }
    else
    {
    char ch;
    while(myfile.get(ch))
    {
    cout<<ch;
    }
    if(myfile.eof())
    {
    cout<<"文件内容已经全部读完"<<endl;
    }
    while(myfile.get(ch))
    {
    cout<<ch;
    }
    }
    system("pause");
    }




    



        C语言提供了格式化输入输出的方法,C++也同样,但是C++的控制符使用起来更为简单方便,在c++下有两中方法控制格式化输入输出。
      1.有流对象的成员函数。
      例如,下列程序以成员函数的方式控制输出的精度:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    int main()
    {
    float pi=3.14159f;
    cout<<pi<<endl;
    cout.precision(2);
    cout<<pi<<endl;
    system("pause");
    }




      2.使用C++输入输出控制符,控制符是在拖文件iomanip.h中定义的对象,与成员函数有一样的效果,控制符不必像成员函数学那样单独调用,它可以直接插入流中使用。
      例如,下列程序以控制符的方式控制输出的精度:
     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    #include <iomanip>
    using namespace std;

    int main()
    {
    float pi=3.14159f;
    cout<<pi<<endl;
    cout<<setprecision(4);
    cout<<pi<<endl;
    system("pause");
    }




      下表我们列出了一些比较常用的控制符号,由于篇幅有限读者请根据自己的需要查阅相关书籍:

    



     

      对于iostream标准库来说包含了众多的成员函数,各函数都有其自身的作用,篇幅问题笔者在这里不能一一说明例举,由于标准输入对象cin提供输入的时候会自动以空格作为分界,给我们获取一行带有空格的完整字符串带来了困难,在这里补充一个非常用有的成员函数----getline()。

      其函数原型为:
      getlin(chiar *str,int size,char='\n');

      第一个参数是字符数组,用于存放整行文本,第二个参数读取的最大字符个数,第三个参数为作为分界界限的字符,默认识是\n,换行符。

    示例代码如下:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    #include <iomanip>
    using namespace std;

    int main()
    {
    char str[100];
    cin.getline(str,sizeof(str),'\n');
    cout<<str<<endl;
    system("pause");
    }




    



        通过上面内容的学习,我们对i/o有了一些基本点基本的认识,现在是该切入正题的时候了,详细学习一下,如何重载左移与右移操作符。

      先说左移(<<)操作符,也就是我们常说的输出操作符。
      对于自定义类来说,重载左移操作符的方法我们常使用类的友元方式进行操作。

    示例代码如下:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    class Test
    {
    public:
    Test(int age = 0,char *name = "\0")
    {
    Test::age = age;
    strcpy(Test::name,name);
    }
    void outmembers(ostream &out)
    {
    out<<"Age:"<<age<<endl<<"Name:"<<this->name<<endl;
    }
    friend ostream& operator <<(ostream& ,Test&);
    protected:
    int age;
    char name[50];
    };
    ostream& operator <<(ostream& out,Test &temp)
    {
    temp.outmembers(out);
    return out;
    }
    int main()
    {
    Test a(24,"管宁");
    cout<<a;
    system("pause");
    }




      上例代码中,我们对void outmembers(ostream &out)的参数使用ostream定义主要是为了可以向它传递任何ostream类对象不光是cout也可以是ofstrem或者是ostrstream和ostringstream类对象,做到通用性。

      重载运算符,我们知道可以是非成员方式也可以是成员方式的,对于<<来说同样也可以是成员方式,但我十分不推荐这么做,因为对于类的成员函数来说,第一个参数始终是会被隐藏的,而且一定是当前类类型的。

      下面的示例代码就是将上面的<<重载函数修改成成员方式的做法:

     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    using namespace std;

    class Test
    {
    public:
    Test(int age = 0,char *name = "\0")
    {
    Test::age = age;
    strcpy(Test::name,name);
    }
    void outmembers(ostream &out)
    {
    out<<"Age:"<<age<<endl<<"Name:"<<this->name<<endl;
    }
    ostream& operator <<(ostream &out)
    {
    this->outmembers(out);
    return out;
    }
    protected:
    int age;
    char name[50];
    };
    int main()
    {
    Test a(24,"管宁");
    a<<cout;
    system("pause");
    }




      从代码实现上,我们将函数修改成了ostream& operator <<(ostream &out),迫不得已将ostream类型的引用参数放到了后面,这是因为,成员方式运算符重载函数第一个参数会被隐藏,而且一定是当前类类型的,这和ostream类型冲突了。由此我们在使用cout输出的时候就必须写成a<<cout;,这样一来代码的可读行就大大降低了,这到底是左移还是右移呢?为此我再一次说明,对于左移和右移运算符的重载是十分不推荐使用成员函数的方式编写的。

      为了巩固学习,下面我们以fstream对象输出为例做一个练习。

    代码如下:
     C++ 代码 //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者
    #include <iostream>
    #include <fstream>
    using namespace std;

    class Test
    {
    public:
    Test(int age = 0,char *name = "\0")
    {
    Test::age = age;
    strcpy(Test::name,name);
    }
    void outmembers(ostream &out)
    {
    out<<"Age:"<<age<<endl<<"Name:"<<this->name<<endl;
    }
    friend ostream& operator <<(ostream& ,Test&);
    protected:
    int age;
    char name[50];
    };
    ostream& operator <<(ostream& out,Test &temp)
    {
    temp.outmembers(out);
    return out;
    }
    int main()
    {
    Test a(24,"管宁");
    ofstream myfile("c:\\1.txt",ios::out,0);
    if (myfile.rdstate() == ios_base::goodbit)
    {
    myfile<<a;
    cout<<"文件创建成功,写入正常!"<<endl;
    }
    if (myfile.rdstate() == ios_base::badbit)
    {
    cout<<"文件创建失败,磁盘错误!"<<endl;
    }
    system("pause");
    }




    



        对于左移运算符重载函数来说,由于不推荐使用成员方式,那么使用非成员方式在类有多重继承的情况下,就不能使用虚函数进行左移运算符重载的区分,为了达到能够区分显示的目的,给每个类分别添加不同的虚函数是必要的。

    示例代码如下:

     C++ 代码
    //程序作者:管宁
    //站点:www.cndev-lab.com
    //所有稿件均有版权,如要转载,请务必著名出处和作者

    #include <iostream>
    #include <fstream>
    using namespace std;

    class Student
    {
    public:
    Student(int age = 0,char *name = "\0")
    {
    Student::age = age;
    strcpy(Student::name,name);
    }
    virtual void outmembers(ostream &out) = 0;
    friend ostream& operator <<(ostream& ,Student&);
    protected:
    int age;
    char name[50];
    };
    ostream& operator <<(ostream& out,Student &temp)
    {
    temp.outmembers(out);
    return out;
    }
    class Academician:public Student
    {
    public:
    Academician(int age = 0,char *name = "\0",char *speciality="\0"):Student(age,name)
    {
    strcpy(Academician::speciality,speciality);
    }
    virtual void outmembers(ostream &out)
    {
    out<<"Age:"<<age<<endl<<"Name:"<<name<<endl<<"speciality:"<<speciality<<endl;
    }
    protected:
    char speciality[80];
    };
    class GraduateStudent:public Academician
    {
    public:
    GraduateStudent(int age = 0,char *name = "\0",char *speciality="\0",char *investigate="\0"):Academician(age,name,speciality)
    {
    strcpy(GraduateStudent::investigate,investigate);
    }
    virtual void outmembers(ostream &out)
    {
    out<<"Age:"<<age<<endl<<"Name:"<<name<<endl<<"speciality:"<<speciality<<endl<<"investigate:"<<investigate<<endl;
    }
    protected:
    char investigate[100];
    };
    int main()
    {
    Academician a(24,"管宁","Computer Science");
    cout<<a;
    GraduateStudent b(24,"严燕玲","Computer Science","GIS System");
    cout<<b;
    system("pause");
    }




      在上面的代码中为了能够区分输出a对象与b对象,我们用虚函数的方式重载了继承类Academician与多重继承类GraduateStudent的outmembers成员函数,由于ostream& operator <<(ostream& out,Student &temp) 运算符重载函数是Student基类的,Student &temp参数通过虚函数的定义可以适应不同派生类对象,所以在其内部调用temp.outmembers(out); 系统可识别不同继类的outmembers()成员函数。

    

    • 标签:
    • iostream
    • c++
    • 作者
    • 输入
    • 对象
    • 代码
    • 成员
    • ostream
    • 函数
    • 程序
    • int
  • 加入的知识群:
    学习元评论 (0条)

    评论为空
    聪明如你,不妨在这 发表你的看法与心得 ~



    登录之后可以发表学习元评论
      
暂无内容~~
顶部