北大C++课后记录:文件读写的I/O流

前言

文件和平常用到的cincout流其实是一回事,可以将文件看作一个有限字符构成的顺序字符流 ,基于此,也可以像cincout读键盘数据那样对文件进行读写。


读写指针

  • 输入流的read指针
  • 输出流的write指针

注:这里的指针并不是普通的内存指针(*ptr),而是一种叫std::streampos类型的指针,是用于在文件流中定位文件位置的数据类型,可以类比成一个int型的数据。


文本文件读写demo

cpp 复制代码
#include <iostream>
#include <fstream>
#include <algorithm>
#include <cstring>
#include <string.h>
#include <vector>
using namespace std;

int main(){
    vector<int> v;
    ifstream srcFile("testin.txt", ios :: in);
    ofstream destFile("testout.txt", ios :: out);
    int t;
    while(srcFile >> t)
        v.push_back(t);
    sort(v.begin(), v.end());
    for(auto i : v)
        destFile << i << ' ';

    //不执行close的话数据还在内存(的缓冲区)里没被磁盘读走
    srcFile.close();
    destFile.close();
    return 0;
}
//testin.txt: 9 84 23 15 56 2 34 123 5 1
//testout.txt: 1 2 5 9 15 23 34 56 84 123 

二进制文件读写demo1


cpp 复制代码
#include <iostream>
#include <fstream>
#include <algorithm>
#include <cstring>
#include <string.h>
#include <vector>
using namespace std;

//写入一个整型数据后接着读出
int main(){
    ofstream fout("some.bat", ios::out | ios::binary);      //从文件头开始,以二进制的形式写入
    int x = 120;
    //从x的内存地址开始传入4字节数据
    //(const char*)强转是为了让编译器拿到(int*)指针后能将其解释成字符型数据
    fout.write((const char*) &x, sizeof(int));
    fout.close();
    int y;
    ifstream fin("some.bat", ios::in | ios::binary);//从文件头开始,以二进制的形式写入
    //从文件开头处取出4字节数据
    //然后将y地址转成(char*)从而在该地址后面放入4字节数据
    fin.read((char*) &y, sizeof(int));
    fin.close();
    cout << y << endl;
    return 0;
}

二进制文件读写demo2

  • 先写入三个对象的数据
cpp 复制代码
#include <iostream>
#include <fstream>
#include <algorithm>
#include <cstring>
#include <string.h>
#include <vector>
using namespace std;
struct Student{
    char name[20];
    int score;
};//24-Byte

int main(){
    Student s;
    ofstream OutFile("students.dat", ios::out|ios::binary);
    while(cin >> s.name >> s.score)
        OutFile.write((char*) &s, sizeof(s));
    OutFile.close();
    return 0;
}
//input:
//Tom 60
//Jack 80
//Jane 40
//^Z Enter
  • 读出并合理地显示内容
cpp 复制代码
#include <iostream>
#include <fstream>
#include <algorithm>
#include <cstring>
#include <string.h>
#include <vector>
using namespace std;
struct Student{
    char name[20];
    int score;
};//24-Byte

int main(){
    Student s;
    ifstream inFile("students.dat", ios::in|ios::binary);
    if(!inFile){        //空流
        cerr << "error" << endl;
        return 0;
    }
    while(inFile.read((char*) &s, sizeof(s))){
        int readBytes = inFile.gcount();	//本次读进来了多少字节数据
        cout << "readBytes: " << readBytes << endl;
        cout << s.name << " " << s.score << endl;
    }
    inFile.close();
    return 0;
}
//output:
//readBytes: 24
//Tom 60
//readBytes: 24
//Jack 80
//readBytes: 24
//Jane 40

二进制文件读写demo3

  • 用一个fstream流对象完成读和写的任务
cpp 复制代码
#include <iostream>
#include <fstream>
#include <algorithm>
#include <cstring>
#include <string.h>
#include <vector>
using namespace std;
struct Student{
    char name[20];
    int score;
};//24-Byte
//Jane------>Mike
int main(){
    Student s;
    fstream iofile("students.dat", ios::in|ios::out|ios::binary);
    if(!iofile){
        cerr << "error" << endl;
        return 0;
    }
    iofile.seekp(2 * sizeof(s), ios::beg);      //从文件头开始将写指针移动两个对象大小的跨度------>到第三个对象的首字节
    //由于Mike和Jane同字符个数,所以写入Mike后'e'的后面已经是'\0'了,其实可以不"+1"让编译器来补终止符。
    iofile.write("Mike", strlen("Mike") + 1);       
    iofile.seekg(0, ios::beg);      //防止读指针随机指到其他地方
    while(iofile.read((char*) &s, sizeof(s))){
        int readBytes = iofile.gcount();
        cout << "readByte: " << readBytes << endl;
        cout << s.name << " " << s.score << endl;
    }
    iofile.close();
    return 0;
}
// output:
// readByte: 24
// Tom 60
// readByte: 24
// Jack 80
// readByte: 24
// Mike 40

总结

  • 文本模式和二进制模式的主要区别在于文本模式会执行一些与文本数据处理相关的特殊转换和处理(比如Windows下所有的'\r\n'会被解释成一个'\n'字符,从而造成数据读取泄漏),而二进制模式不会执行这些操作,二进制模式保留原数据。
  • 这里写入、读出时总利用char*作为参数的其中之一作用:是为了更好地观察、理解数据,否则其他数据类型打开在用户视角是一片乱码(可以用记事本打开dat文件验证)。
相关推荐
娅娅梨26 分钟前
C++ 错题本--not found for architecture x86_64 问题
开发语言·c++
兵哥工控31 分钟前
MFC工控项目实例二十九主对话框调用子对话框设定参数值
c++·mfc
汤米粥32 分钟前
小皮PHP连接数据库提示could not find driver
开发语言·php
冰淇淋烤布蕾35 分钟前
EasyExcel使用
java·开发语言·excel
我爱工作&工作love我38 分钟前
1435:【例题3】曲线 一本通 代替三分
c++·算法
拾荒的小海螺41 分钟前
JAVA:探索 EasyExcel 的技术指南
java·开发语言
马剑威(威哥爱编程)1 小时前
哇喔!20种单例模式的实现与变异总结
java·开发语言·单例模式
娃娃丢没有坏心思1 小时前
C++20 概念与约束(2)—— 初识概念与约束
c语言·c++·现代c++
lexusv8ls600h1 小时前
探索 C++20:C++ 的新纪元
c++·c++20
lexusv8ls600h1 小时前
C++20 中最优雅的那个小特性 - Ranges
c++·c++20