您的位置:首页 > 编程语言 > C语言/C++

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

2014-11-04 10:25 190 查看
  本节对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中可以连续使用“>>”或“<<”运算符,与所有“>>”和“<<”重载函数一致。第一个参数是输入或输出流的引用,作为“>>”或“<<”的左操作数;第二个参数为用户定义类的引用,作为右操作数,流用作函数参数,必须是引用调用,不能是传值调用。

[b]  示例3 重载插入运算符“<<”[/b]

#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;
}


[b]示例 4 用户定义的复数类Complex的输入与输出。[/b]

#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;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: