C语言的输入和输出
C语言中我们用到的最频繁的输入输出方式就是scanf ()与printf()。 scanf(): 从标准输入设备(键 盘)读取数据,并将值存放在变量中。printf(): 将指定的文字/字符串输出到标准输出设备(屏幕)。 注意宽度输出和精度输出控制。C语言借助了相应的缓冲区来进行输入与输出。如下图所示:
流是什么?
“流”即是流动的意思,是物质从一处向另一处流动的过程,是对一种有序连续且具有方向性的数 据( 其单位可以是bit,byte,packet )的抽象描述。 C++流是指信息从外部输入设备(如键盘)向计算机内部(如内存)输入和从内存向外部输出设 备(显示器)输出的过程。这种输入输出的过程被形象的比喻为“流”。 它的特性是:有序连续、具有方向性。
为了实现这种流动,C++定义了I/O标准类库,这些每个类都称为流/流类,用以完成某方面的功 能
C++IO流
C++提供了四个全局流对象
cin
、cout
、cerr
、clog
。cin:读取数据。
cout:输出数据。
cerr:输出错误。
clog:输出日志。
cin和cout可以直接输入和输出内置类型数据,因为标准库已经将所有内置类型的输入和输出全部重载了,如果要输出自定义类型,只需要在类当中重载
operator <<
和operator>>
即可。
循环读取数据
// 单个元素循环输入 while(cin>>a) { // ... } // 多个元素循环输入 while(cin>>c>>a>>b>>c) { // ... } // 整行接收 while(cin>>str) { // ... }
VS编译器下:Ctrl+Z可以结束循环读入。
while(cin>>a)
,这里调用的是operator >>
返回值是istream
类型的对象,然后while()
循环会去判断返回值的真假情况,所以又会去调用operator bool
。
当输入Crtl+Z的时候就是返回
false
的时候。
#include<iostream> using namespace std; class Date { friend istream& operator>>(istream& in, Date& d1); friend ostream& operator<<(ostream & out, Date& d1); public: Date(int year = 2023, int month = 8, int days = 17) :_year(year) , _month(month) , _days(days) {} operator bool() { if (_year == 0) return false;//随便写的 else return true; } private: int _year; int _month; int _days; }; istream& operator>>(istream& in, Date& d1) { return in >> d1._year >> d1._month >> d1._days; } ostream& operator<<(ostream & out, Date& d1) { return out << d1._year << d1._month << d1._days; } int main(void) { Date d1(0, 1, 1); while (d1) { cout << "while(d1)" << endl; } cout << "false" << endl; return 0; } =============================================== 输出结果为: false
C++文件IO流
读写文件
struct ServeInfo { char _address[32]; int _port; }; class ConfigManager { public: ConfigManager(const char* filename) :_filename(filename) {} void WriteBin(const ServeInfo& info) { ofstream ofs(_filename, ios_base::out | ios_base::binary); ofs.write((const char*)&info, sizeof(info)); //ofs << info._address <<endl<< info._port << endl; } void ReadBin(ServeInfo& info) { ifstream ifs(_filename, ios_base::in | ios_base::binary); ifs.read((char*)&info, sizeof(info)); //ifs >> info._address >> info._port>>info._date; } private: string _filename; }; int main(void) { ServeInfo winfo= { "192,168.1.1", 8888 };//初始化信息 ConfigManager cf_bin("test.txt");//传递文件地址 cf_bin.WriteBin(winfo);//传递信息,写入文件 ServeInfo rbinfo; cf_bin.ReadBin(rbinfo);//打印文件内容 cout << rbinfo._address << " " << rbinfo._port << " " << endl; return 0; }
stringstream
全都转换为string
将数值类型数据格式转换为字符串
int a = 12345678910; string sa; stringstream s; s << a; s >> sa; cout << typeid(s).name() << endl; // clear() // 注意多次转换时,必须使用clear将上次转换状态清空掉 // stringstreams在转换结尾时(即最后一个转换后),会将其内部状态设置为badbit // 因此下一次转换是必须调用clear()将状态重置为goodbit才可以转换 // 但是clear()不会将stringstreams底层字符串清空掉 // s.str(""); // 将stringstream底层管理string对象设置成"", // 否则多次转换时,会将结果全部累积在底层string对象中 s.str(""); s.clear();
clear() 注意多次转换时,必须使用clear将上次转换状态清空掉 stringstreams在转换结尾时(即最后一个转换后),会将其内部状态设置为badbit 因此下一次转换是必须调用clear()将状态重置为goodbit才可以转换 但是clear()不会将stringstreams底层字符串清空掉 s.str(""); 将stringstream底层管理string对象设置成"", 否则多次转换时,会将结果全部累积在底层string对象中
字符串拼接
stringstream sstream; sstream << "first" << " " << "string"; sstream << " second string"; cout << "strResult: " << sstream.str() << endl;
序列化和反序列化结构数据
ServeInfo info= { "123.123.123.123",80 }; ostringstream oss; oss << info._address << " " << info._port << endl; string str = oss.str(); cout << str << endl; ServeInfo rinfo; istringstream iss(str); iss >> rinfo._address >> rinfo._port; cout << "==========================================" << endl; cout << rinfo._address << " " << rinfo._port << endl;
- stringstream实际是在其底层维护了一个string类型的对象用来保存结果。
- 多次数据类型转化时,一定要用clear()来清空,才能正确转化,但clear()不会将 stringstream底层的string对象清空。
- 可以使用s. str(“”)方法将底层string对象设置为""空字符串。
- 可以使用s.str()将让stringstream返回其底层的string对象。
- stringstream使用string类对象代替字符数组,可以避免缓冲区溢出的危险,而且其会对参 数类型进行推演,不需要格式化控制,也不会出现格式化失败的风险,因此使用更方便,更安全。