北大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文件验证)。
相关推荐
蜀黍@猿11 分钟前
C/C++基础错题归纳
c++
古希腊掌管学习的神19 分钟前
[LeetCode-Python版]相向双指针——611. 有效三角形的个数
开发语言·python·leetcode
赵钰老师19 分钟前
【R语言遥感技术】“R+遥感”的水环境综合评价方法
开发语言·数据分析·r语言
雨中rain26 分钟前
Linux -- 从抢票逻辑理解线程互斥
linux·运维·c++
就爱学编程27 分钟前
重生之我在异世界学编程之C语言小项目:通讯录
c语言·开发语言·数据结构·算法
Oneforlove_twoforjob1 小时前
【Java基础面试题025】什么是Java的Integer缓存池?
java·开发语言·缓存
emoji1111111 小时前
前端对页面数据进行缓存
开发语言·前端·javascript
每天都要学信号1 小时前
Python(第一天)
开发语言·python
TENET信条1 小时前
day53 第十一章:图论part04
开发语言·c#·图论
生信圆桌1 小时前
【生信圆桌x教程系列】如何安装 seurat V5版本R包,最详细安装手册
开发语言·r语言