读取文件大体上可以分为五步(之所以用in,读取文件这个行为对于程序来说相当于输入数据,所以用in,写文件的行为,是数据从程序中流出,所以用out。in,out都是相对于程序而言)
- 声明头文件:#include<fstream>
- 实例化对象:ifstream fin ,ifstream是<fstream>中的一个类,fin是一个实例化对象,之所以起这个名字是类比cin,实际上他们有很多相似的地方
- fin.open("文件名”,"打开方式"), 本文的打开方式采用ios::in , 以文本的方式写,同时还要利用fin.is_open()判断是否成功打开了这个文件,这个函数的返回值类型是bool。
- 开始执行写文件的操作
- fin.close();
其中除了第四步都是固定流程,第四步可以有四种方法。
第一种方法:
void test02() { ifstream fin; fin.open("记录本.txt", ios::in); if (!fin.is_open()) { cout << "无法找到这个文件!" << endl; return; } char buff[1024] = { 0 }; while (fin >> buff) { cout << buff<<endl; } fin.close(); }
类似于cin>>写入,fin >> buff遇到EOF(end of file)会返回false,跳出while循环,遇到EOF之前,每执行一次fin >> buff,就会将buff填满上次终止条件和下次终止条件之间的内容,然后执行cout << buff<<endl;按行输出。
不过这种方法有一个巨大的问题,那就是fin和cin一样会将所有的空格当做终止条件,这样就没有办法输出空格信息了(因为空格都被endl回车代替了)
比如文件中的内容是:
大家好 我是渣渣辉!
今晚 八点不见不散
我在 贪玩蓝月等你
每一行都有一个空格,那么输出的结果就是:
大家好
我是渣渣辉!
今晚
八点不见不散
我在
贪玩蓝月等你
程序结束了
显然多数情况下,这都不是我们想要的输出结果,我们想要的输出结果应该是和文本内容相同的格式。
第二种方法:
利用<string>中的getline函数,按行读取,这样每一行内的所有空格也可以顺利读取
void test02() { ifstream fin; fin.open("记录本.txt", ios::in); if (!fin.is_open()) { cout << "无法找到这个文件!" << endl; return; } string buff; while (getline(fin, buff)) { cout << buff<<endl; } fin.close(); }
结果:
大家好 我是渣渣辉!
今晚 八点不见不散
我在 贪玩蓝月等你
非常好,和文件中的内容完全一致!我比较喜欢这种方法
第三种:
利用fin对象内部的getline方法,这个名字虽然和第二种中的一样,但是传入的参数不同,并不是同一个函数。
void test02() { ifstream fin; fin.open("记录本.txt", ios::in); if (!fin.is_open()) { cout << "无法找到这个文件!" << endl; return; } char buff[1024] = { 0 }; while (fin.getline(buff, sizeof(buff))) { cout << buff<<endl; } fin.close(); }
结果和第二种完全相同;
第四种:
最暴力的一种把文件中的字符一个一个读取出来,在输出到控制台上,不用考虑回车换行的问题,因为文件中的\n被读取出来之后输出到控制台上自动就换行了,利用fin.get()函数,这个函数的返回值是它读取到的字符,当遇到EOF的时候返回EOF,通过判断跳出while循环
void test02() { ifstream fin; fin.open("记录本.txt", ios::in); if (!fin.is_open()) { cout << "无法找到这个文件!" << endl; return; } char buff=0; while ((buff = fin.get())!=EOF) { cout << buff; } fin.close(); }
结果和第二种第三种一样:
大家好 我是渣渣辉!
今晚 八点不见不散
我在 贪玩蓝月等你
总结:第一种方法有无法读取空格的缺陷,当文件中有空格的时候应该避免使用第一种方法,第二种方法和第三种方法主要思想是按行读取,只不过一个是string库中的函数,一个是istream中的函数,参数不同,第四种是暴力方法,把每个字符读出来逐个输出。