VC++ 之 输入/输出类库(二)

  本节对cin,cout,cerr,clog,>>和<<(提取和插入运算符)的使用细节作进一步讨论。

提高标准输入/输出的健壮性

◆ 1、标准设备输入使用要点

  • cin为缓冲流。键盘输入的数据保存在缓冲区中,当要提取时,是从缓冲区中拿。如果一次输入过多,会留在那儿慢慢用,如果输入错了,必须在回车之前修改,如果回车键按下就无法挽回了。只有把输入缓冲区中的数据取完后,才要求输入新的数据。不可能用刷新来清除缓冲区,所以不能输错,也不能多输!
  • 输入的数据类型必须与要提取的数据类型一致,否则出错。出错只是在流的状态字state(枚举类型io_state)中对应位置位(置1),程序继续。所以要提高健壮性,就必须在编程中加入对状态字state的判断。
  • 空格和回车都可以作为数据之间的分格符,所以多个数据可以在一行输入,也可以分行输入。但如果是字符型和字符串,则空格(ASCII码为32)无法用cin输入,字符串中也不能有空格。回车符也无法读入。
  • 输入数以后再输入字符或字符串:如果数后直接加回车,应该用cin.get()提取回车。如果还有空格,则要清空缓冲区。

◆ 2、程序运行状态
状态字state为整型,其的各位在ios中说明:
enum ios_state
{
       goodbit=0x00, //流正常
       eofbit=0x01,    //输入流结束忽略后继提取操作;或文件结束已无数据可取
       failbit=0x02, //最近的I/O操作失败,流可恢复
       badbit=0x04, //最近的I/O操作非法,流可恢复
       hardfail=0x08     // I/O出现致命错误,流不可恢复,VC6.0++不支持
}

读取状态的有关操作如下:
inline int ios::rdstate() const //读取状态字
{return state;}

inline int ios:operator!() const //可用操作符!()代替fail()
{return state&(badbit|failbit);}

inline int ios::bad() //返回非法操作位
{ return state & badbit;}

inline void ios::clear(int _i) //人工设置状态,可用来清状态
{ lock();state=_i;unlock();}

inline int ios::eof() const //返回流(文件)结束位
{return state&eofbit;}

inline int ios::fail() const //返回操作非法和操作失败这两位
{return state&(badbit|failbit);}

inline int ios::good() const //正常返回1,否则返回0
{return state==0;}

  示例 1提高输入的健壮性。输入时需要故意输错,以测试健壮性。
#include<iostream>
using namespace std;
int main(){
    char str[256];
    int i;
    cout<<"请输入整数:"<<endl;//强制清空缓冲区,保证输出,不会等缓冲区溢出再输出
    cin>>i;//可故意输入若干非数字字符,下次再输入若干字符加数字串加若干非数字字符进行检测
    while(cin.fail()){
        cout<<"状态字为:"<<cin.rdstate()<<endl;
        cin.clear(0);
        cin.getline(str,255);//读空缓冲区
        cout<<"输入错误,请重新输入整数"<<endl;
        cin>>i;
    }
    cin.getline(str,256);//读空缓冲区
    cout<<"请输入字符串"<<endl;
    cin.getline(str,255);//B行
    cout<<"输入整数为:"<<i<<endl;
    cout<<"输入字符串为:"<<str<<endl;
    return 0;
}

标准输入/输出成员函数

 1、输入流成员函数声明
(1)字符输入:
    int istream::get();
    //提取一个字符,包括空格,制表,backspace和回车等,
    //与cin有所不同.注意返回为整型
    istream&istream::get(char &);
    istream&istream::get(unsigned char &);
提取一个字符,放在字符型变量中

(2)字符串输入:
    istream&istream::get(char *,int,char=’\n’);
    istream&istream::get(unsigned char *,int,char=’\n’);
    istream&istream::getline(char *,int,char=’\n’);
    istream&istream::getline(unsigned char *,int,char=’\n’);
提取的串放在第一个参数为开始地址的存储区(不查边界);第二个参数为至多提取的字符个数(指定为n,最多取n-1个,再加一个字符串结束符);第三个参数为结束字符,遇此字符则结束,默认为回车换行符。

get系列函数要求单独提取结束字符。getline提取字符串时如遇到指定结束符则提取该结束符,但不保存在串中。这两个函数都会在提取的一系列字符后加一个串结束符,返回值为对象本身(*this)。

(3)其他函数:
函数gcount()返回最后一次提取的字符数量,包括回车:
    int istream::gcount();

函数ignore()读空(指定一个大的数量)缓冲区:
    istream&istream::ignore(int=1,int=EOF);
第一个参数为要提取的字符数量,默认为1;第二个参数为结束字符,提取该结束字符,但对所提取的字符不保存不处理,作用是空读。第二个参数的默认值EOF为文件结束标志。

在iostream中EOF定义为-1,在int get()函数中,读入输入流结束标志Ctrl+Z(^Z)时,函数返回EOF,为了能表示EOF的“-1”值,返回类型为int。采用cin.eof()函数,当前所读为EOF则返回非零,注意函数自身未从流中读取。

  示例2: ignore()和gcount()函数使用。
#include<iostream>
#include<cstring>
using namespace std;

int main(){
    char str[255];
    int i,n;
    cout<<"输入字符"<<endl;        //输入^Z,一旦输入^Z全部结束,不能输入其它字符
    i=cin.get();
    cout<<endl;
    n=cin.rdstate();                         //读取状态字
    cout<<"状态字为:"<<n<<endl;             //状态字为1,流结束
    cout<<"当输入字符时,取得的是:"<<i<<endl; //-1,输入^Z时,返回EOF,即-1
    if(n==0) cin.ignore(255,'\n');           //清除多余的字符和回车符
    cin.clear(0);                            // A    使流恢复正常
    cout<<"输入字符串1:"<<endl;
    cin.getline(str,255);
    cout<<endl;
    cout<<"状态字为:"<<cin.rdstate()<<endl;
    i=cin.gcount();
    cout<<"字符串为:"<<str<<'\t'<<"读入字符数为:"<<i<<'\t';
    cout<<"串长为:"<<strlen(str)<<endl;
    cin.clear(0);                            // A    使流恢复正常
    cout<<"输入字符串2:"<<endl;
    cin.getline(str,255);
    cout<<endl;
    cout<<"状态字为:"<<cin.rdstate()<<endl;
    i=cin.gcount();
    cout<<"字符串为:"<<str<<'\t'<<"读入字符数为:"<<i<<'\t';
    cout<<"串长为:"<<strlen(str)<<endl;
    return 0;
}

  2、输出流成员函数声明
    ostream&ostream::put(char);
    //输出参数字符
    ostream&ostream::put(unsigned char);
    ostream&ostream::put(signed char);
    ostream&ostream::flush();
    //刷新一个输出流,用于cout和clog

重载插入和提取运算符

重载必须保留原来的使用特性。重载只能在用户定义类中,将重载的运算符的函数说明为该类的友元函数:
    friend istream&operator>>(istream&,className&);
    friend ostream&operator<<(ostream&,className&);
函数的返回值是对输入或输出流的引用,这是为了保证在cin和cout中可以连续使用“>>”或“<<”运算符,与所有“>>”和“<<”重载函数一致。第一个参数是输入或输出流的引用,作为“>>”或“<<”的左操作数;第二个参数为用户定义类的引用,作为右操作数,流用作函数参数,必须是引用调用,不能是传值调用。

  示例3 重载插入运算符“<<”

 

#include<iostream>
using namespace std;

template <typename T>struct Node{
    T  key;
    // 其他域省略
};//再次指出分号不可少
template <typename T,int size>class Orderedlist{
    int maxsize;
    int last;
    Node<T> slist[size];
public:
    Orderedlist(){last=-1;maxsize=size;}
    void BubbleSort();
    bool Insert(Node<T> & elem,int i);
    void print();
    // 无关成员函数省略
};//再次指出分号不可少
template <typename T,int size> bool Orderedlist<T,size>::Insert(Node<T> & elem ,int i){
    if (i<0||i>last+1||last==maxsize-1) return false;
    else{
        for (int j=last;j>i;j--) slist[j]=slist[j-1];
        slist[i]=elem;
        last++;
        return true;
    }
}
template <typename T,int size> void Orderedlist<T,size>::print(){
    int i;
    for(i=0;i<=last;i++){
        cout<<slist[i].key;
        if(i%5==4) cout<<endl;
    }
    cout<<endl;
}
template <typename T,int size> void Orderedlist<T,size>::BubbleSort(){
    bool noswap;
    int i,j;
    Node<T> temp;
    for (i=0;i<last;i++){//最多做n-1趟
        noswap=true;    //未交换标志为真
        for(j=last;j>i;j--){//从下往上冒泡
            if(slist[j].key<slist[j-1].key){
                temp=slist[j];
                slist[j]=slist[j-1];
                slist[j-1]=temp;
                noswap=false;
            }            
        }
        if(noswap) break;    //本趟无交换,则终止算法。
    }
}

class mystring{
    char str[20];
    int maxsize;
    int last;
public:
    mystring(){
        last=-1;
        maxsize=20;
        str[0]='\0';
    }
    mystring(char *s){//本例为了简化,健壮性并不好
        last=-1;
        maxsize=20;
        do{
            last++;
            str[last]=s[last];
        }while(s[last]!='\0');
    }
    ~mystring(){}
    friend ostream & operator<<(ostream & ,const mystring &);//流类作为形式参数必须是引用
    bool operator<(mystring &);
    mystring & operator=(char * ms);
};
bool mystring::operator<(mystring & ms){//重载<运算符
    int i=0,k;
    do{
        k=str[i]-ms.str[i];
        i++;
    }while(k==0&&i<last&&i<ms.last);
    if(k<0) return true;
    if(i==last&&i!=ms.last) return true;
    return false;
}
ostream & operator<<(ostream & s,const mystring & cstr){
    return s<<cstr.str<<'\t';
}
mystring & mystring::operator=(char * ms){
    last=-1;    
    do{
        last++;
        str[last]=ms[last];
    }while(ms[last]!='\0'&&last<maxsize-1);
    ms[last]='\0';
    return *this;
}

int main(){
    const int h=10;
    int i;
    Orderedlist<mystring,100> ordlist;
    char mslist[h][5]={"cat","book","car","zoo","fish","cab","dog","cap","fox","can"};
    Node<mystring> n[h];//定义结点数组
    for(i=0;i<h;i++)  n[i].key=mslist[i];// 结点数组赋值
    for(i=0;i<h;i++)  ordlist.Insert(n[i],i); //建立顺序表
    cout<<"未排序表:"<<endl;
    ordlist.print();
    ordlist.BubbleSort();
    cout<<"已排序表:"<<endl;
    ordlist.print();
    return 0;
}

 

示例 4 用户定义的复数类Complex的输入与输出。
#include<iostream>
using namespace std;

class Complex{
    double Real,Image;
public:
    Complex(double r=0.0, double i=0.0):Real(r),Image(i){}//定义构造函数
    friend ostream&operator<<(ostream&s,const Complex&z);
    friend istream&operator>>(istream&s,Complex&a);
};
ostream&operator<<(ostream&s,const Complex &z){     //流类作为形式参数必须是引用
    return s<<'('<<z.Real<<','<<z.Image<<')';
}
istream&operator>>(istream&s,Complex &a){//格式为d,(d),(d,d)
    double re=0,im=0;
    char c=0;
    s>>c;//是否由括号开始
    if(c=='('){
        s>>re>>c;//实部
        if(c==',')s>>im>>c;//虚部
        if(c!=')')s.clear(ios::failbit);//漏了括号给一个操作失败标志
    }
    else{
        s.putback(c);//无括号,返回一个字符到输入缓冲区
        s>>re;//实数
    }
    if(s)a=Complex(re,im);
    return s;
}
int main(){
    Complex a,b,c;
    cout<<"输入一个实数"<<endl;
    cin>>a;
    cout<<"输入一个用括号括起来的实数"<<endl;
    cin>>b;
    cout<<"输入一个用括号括起来复数"<<endl;
    cin>>c;
    cout<<"a="<<a<<'\t'<<"b="<<b<<'\t'<< "c="<<c<<'\n';
    return 0;
}

 

posted @ 2014-11-04 10:25  Delphi爱好者2014  阅读(473)  评论(0编辑  收藏  举报