cpp文件操作

news/2024/7/20 23:11:23 标签: c++, ios

文件操作

数据流

在cpp中,流(stream)是一个抽象概念,用于描述如何从一个位置到又一个位置传输数据。流主要用于I/O操作。

数据流包括两大类:1. 输入流(istream):数据从某个源流入程序, 2. 输出流(ostream):数据从程序流向某个目的地。

根据操作对象不同分为文件流、字符串流、控制台流。

控制台流

控制台流(<iostream>)主要用于标准输入输出操作的一部分。这些流提供了一个接口,用于与用于进行交互,即从键盘读取和向屏幕输出,无论程序在那种操作系统或平台上运行。

iostream:同时包括istreamosteam,是一种多继承的关系,在程序应用中可以只包含头文件<iostream>

常见对象有:cin(标准输入),cout(标准输出),cerr(用于显示错误信息)。

<<>>运算符,是ostream类使用友元函数的方式对各种基本数据类型重载了<<操作符,使我们能够方便地将数据输出到流中。

文件流

文件流主要包括输入文件流<ifstream>输出文件流<ofstream>输入输出文件流<fstream>,这些类都定义在<fstream>头文件中。

与标准输入输出流cincout不同,文件流需要指定具体的文件来操作。因此,我们需要创建一个流对象并为其指定一个文件名和其他相关属性。

ofstream类的默认构造函数原形为:

ofstream::ofstream(const char *filename, int mode = ios::out, int pmode = filebuf::openprot);

其中filmname表示指定要打开的文件名。

mode是一个标志位,用于指定如何打开文件。默认值是ios::out,意味着文件以输出模式打开

pmode是一个与平台相关的权限参数,指定了文件的权限。

类型解释
ios::app输出追加模式。当写入文件时,数据会被追加到文件的末尾,而不是覆盖文件的内容。
ios::ate初始位置在文件的末尾,但你可以移动到文件中的任何位置进行读写操作。
ios::binary文件以二进制模式打开。不使用这个模式时,默认为文本模式。
ios::in文件以输入模式打开(通常与 ifstream 一起使用)。
ios::out文件以输出模式打开(默认值)。
ios::trunc如果文件已存在,其内容将被截断或删除。
ios:nocreate不建立文件,所以文件不存在时打开失败
ios:.noreplace不覆盖文件,打开文件时如果文件存在失败

打开文件属性值

类型解释
0普通文件,打开访问
1只读文件
2隐含文件
4系统文件

“或”或者“+”把以上属性连接起来,如3或1|2就是以只读和隐含属性打开文件。  文件使用完后可以使用close成员函数关闭文件。

ios::app,当你用这种模式打开文件时,先检查文件的状态。例如,你可以使用 file.is_open() 来确定文件是否成功打开,以及使用 file.fail() 来检查是否有任何I/O错误。这有助于确保文件操作的安全和有效。

字符串流

字符串流定义在<sstream>头文件中,主要有以下三种类型的字符串流:

  1. istringstream: 输入字符串流,允许从一个 std::string 对象中读取。
  2. ostringstream: 输出字符串流,允许写入到一个 std::string 对象。
  3. stringstream: 同时支持读写操作的字符串流。

文件处理

C++的文件处理也可以看作为一个对象,使用文件流的类,使用头文件<fstream>,处理的时候有文本文件二进制文件之分,主要的区别就是存储的形式。

文件处理可以概括成如下几个过程:

  1. 包含头文件
  2. 创建文件流对象
  3. 打开文件
  4. 文件读写
  5. 关闭文件
打开文件

打开文件的核心:打开文件涉及到通知操作系统你希望访问文件的特定部分(例如读、写或两者兼而有之),并且通常会得到一个文件句柄或文件描述符,该描述符后续可用于读写操作。

下面是open()函数的标准语法,open()ifstreamofstreamfstream类的成员函数,用于打开文件。

void open(const char *filename, ios::openmode mode = ios::in | ios::out);

open() 成员函数的第一参数指定要打开的文件的名称和位置,第二个参数定义文件被打开的模式。

image-20231010153854667

以上的不同模式标志,可以组合使用为文件指定复杂的打开行为。

例如:

  • 写入模式与截断模式
ofstream outf;
outf.open("file.txt", ios::out | ios::trunc);

ios::trunc:当与ios::out一起使用时,这个标志表示如果文件已经存在,则其内容应被截断或清空。如果不指定这个标志并且文件已存在,写入操作通常会覆盖文件的现有内容,从文件的开头开始。

  • 读写模式
fstream file;
file.open("file.txt", ios::out | ios::in);

结合ios::inios::out,意味着你可以对文件进行读写操作。当使用这种模式打开文件时,现有的文件内容不会被自动截断或清空。你可以在文件中任意地移动,进行读和写操作。

文件的写入与输出
  1. 写入文件

在cpp中,可以使用流插入运算符(<<)将数据写入文件,类似于将信息输出到控制台的方式。

对比控制台输出和文件输出:

std::cout << "Hello, World!";

这是cout代表标准输出的ostream类实例。

std::ofstream outFile("filename.txt");
outFile << "Hello, File!";

outFileofstream 类的实例。 ofstream 代表“输出文件流”,允许将数据写入文件。

  1. 读取文件

从文件读取涉及流提取运算符 ( >> ),类似于从键盘获取输入。唯一不同的是,在这里使用的是 ifstream 或 fstream 对象,而不是 cin 对象。

  1. 关闭文件

在cpp程序终止时,他会自动关闭并且刷新所有流,释放所有分配的内存,并关闭所有打开的文件,但显示关闭文件是一个好习惯。

使用完文件后关闭文件的原因如下

  • To make sure that all buffered data gets written.确保所有缓冲数据都被写入。
  • To release the resources associated with the file.释放与该文件关联的资源。
  • To prevent any potential data corruption.防止任何潜在的数据损坏。

下面是close()函数的标准语法,close() 函数是 fstream、ifstream 和ofstream 对象的一个成员。

void close()

文件写入简单示例:

#include<iostream>
#include<fstream>
using namespace std;

int main(){
    // 创建流对象
    ofstream outf;
    // 打开文件
    outf.open("file.txt", ios:: app);
    // 写入数据
    outf << "Hello World" << endl;
    // 关闭文件
    outf.close();
    return 0;
}

文件读取的简单示例:

#include<iostream>
#include<fstream> //文件流的头文件
using namespace std;
int main()
{
    char a;
    ifstream infile; //定义ifstream类(输入文件流类)对象outfile
    infile.open("file.txt");//打开文件 使文件流与c++.txt文件建立关联
    while (!infile.eof())
    {
        infile.get(a); //依次获取文件中每个字符 并输出
        cout << a;
    }
    infile.close(); //关闭文件 使文件流与c++.txt文件断开关联
}

这种方法效率低,不建议使用。

第二种方法:

char buf[1024] = {0};
while (ifs >> buf) {
    cout << buf << endl;
}

第三种:

char buf[1024] = {0};
while (ifs.getline(buf, sizeof(buf))) {
    cout << buf << endl;
}

第四种:

string buf;
while (getline(ifs, buf)) {
    cout << buf << endl;
}
ifs.close();
二进制文件读写

二进制写文件主要利用流对象调用成员函数write(),这个函数直接从内存中的某个位置(由buffer指针指定)写入指定的字节数(len)到文件中。

函数原型:ostream& write(const char*buffer,int len);

#include<iostream>
#include<fstream> //文件流的头文件
using namespace std;
class Student {
public:
    Student() {}
    Student(string name,int age) :name(name), age(age) {}
protected:
    string name;
    int age;
};
int main() {
    ofstream bof;
    Student s1("张三", 100);
    bof.open("student.txt", ios::out|ios::binary);
    bof.write((const char*) & s1, sizeof(Student));
    bof.close();
    return 0;
}

读取文件用read 函数从文件中读取指定的字节数并将其放入指定的内存位置。

ifstream bif;
    bif.open("student.txt", ios::in | ios::binary);
    if (!bif.is_open()) {
        cout << "文件打开失败" << endl;
        return 0;
    }
    Student s2;
    bif.read((char*)&s2, sizeof(Student));
    cout << s2.name << " " << s2.age << endl;
    bif.close();

fstream文件操作

打开文件和关闭文件

打开文件fstream可以在生命流对象时传入文件名打开文件,也可以使用open()函数打开文件。

文件打开后必须关闭,fstream提供了close()函数关闭文件。

打开文件

使用构造函数声明对象时打开文件:

fstream ioFile("data.txt", ios::in | ios::out);

使用open()函数打开文件:

fstream file;
file.open("example.txt", ios::in | ios::out);

如果只传文件名,系统会自动根据文件类型选择默认的打开方式。

  • ios::in:读取模式
  • ios::out:写入模式
  • ios::app:追加模式。数据会被写入文件的末尾而不是覆盖已有内容。
  • ios::trunc:如果文件存在,则先删除文件的内容,然后再打开它。
  • ios::binary:以二进制模式打开文件。

模式和属性可以单独使用,也可以混合使用。在混合使用时,需要使用逻辑连接符或|链接。

ios::out会默认清空文件,即ios::out|ios::trunc打开文件的同时清空文件。如果不想清空文件,那么设置读写模式为ios::out|ios::app,以这种模式打开文件后,是以追加内容的方式写入文件。

fstream提供了多种读写操作,包括:

  • <<>>:这些操作符用于向文件写入数据和从文件读取数据。
  • read()write():这些函数用于处理二进制文件。
  • put()get():这些函数分别用于写入和读取单个字符。
  • getline():这个函数用于从文件中读取一行文本。

示例:

读取文本文件:

ifstream file("example.txt");
string line;
while(getline(file, line)) {
    cout << line << endl;
}
file.close();

读取二进制文件:

ifstream file("binaryExample.bin", ios::binary);
int number;
file.read((char*)&number, sizeof(number));
file.close();
  • 文本文件:这些文件主要包含文本,如.txt。你通常会使用逐行读取/写入的方式来处理它们。
  • 二进制文件:这些文件包含二进制数据,如图像、音频或编译后的程序。当处理这些文件时,你会使用块读写的方式,这意味着你可能会一次读取/写入多个字节。

完整示例:

#include <iostream>
#include <fstream>  // 1. 包含头文件
#include <string>
using namespace std;

int main() {
    string str;
    fstream iofile;  // 2. 创建对象

    // 3. 打开文件并清空内容
    iofile.open("file.txt", ios::out | ios::in | ios::trunc);

    // 4. 写入数据
    iofile << "这里是写入内容测试" << endl;
    iofile << "this is test" << endl;
    iofile.close();  // 5. 关闭文件

    cout << "写入完毕" << endl;

    // 重新打开文件进行读取
    iofile.open("file.txt", ios::out | ios::in);
    while (getline(iofile, str))  // 循环读取
    {
        cout << str << endl;
    }
    iofile.close();

    cout << "读取完毕" << endl;
    return 0;
}

二进制数据读写文件:

#include <iostream>
#include <fstream>
using namespace std;

int main() {
    int numbers[] = {10, 20, 30, 40, 50};
    fstream iofile;

    // 1. 打开文件以进行二进制写入
    iofile.open("binaryfile.bin", ios::out | ios::binary | ios::trunc);
    iofile.write((char*) &numbers, sizeof(numbers));
    iofile.close();

    cout << "数据已写入二进制文件" << endl;

    int readNumbers[5] = {0};

    // 2. 打开文件以进行二进制读取
    iofile.open("binaryfile.bin", ios::in | ios::binary);
    iofile.read((char*) &readNumbers, sizeof(readNumbers));
    iofile.close();

    cout << "从二进制文件读取的数据:" << endl;
    for(int i = 0; i < 5; i++) {
        cout << readNumbers[i] << " ";
    }
    cout << endl;

    return 0;
}

文件定义与大小

当你在处理文件时,经常需要控制文件的读写位置或查询其大小。C++提供了一组非常强大的工具来实现这一目的,下面将详细介绍这些功能及其用法。

文件定位:seekg() 和 seekp()

seekg()seekp() 分别用于设置输入和输出流的位置。

  • seekg():设置输入流的位置(对应读操作)。
  • seekp():设置输出流的位置(对应写操作)。

这些函数需要两个参数:

  • offset:这是从给定起点移动的字节量。它可以是正数(向前移动)或负数(向后移动)。
  • origin:这定义了从哪里开始计算偏移。常用的起点有:
    • ios::beg:从文件开始处。
    • ios::cur:从文件的当前位置。
    • ios::end:从文件的末尾。
示例:
fstream file("example.txt", ios::in | ios::out);

// 将读指针从文件开始处向后移动2个字节
file.seekg(2, ios::beg);

// 将写指针从当前位置向后移动2个字节
file.seekp(2, ios::cur);
2. 获取文件大小:tellg() 和 tellp()

这两个函数返回当前的读或写位置,通常表示为从文件开始到当前位置的字节数。

  • tellg():获取输入流的当前位置。
  • tellp():获取输出流的当前位置。

如果你想知道整个文件的大小,一个常见的方法是将读指针移动到文件的末尾,然后使用 tellg()

示例:
fstream file("example.txt", ios::in | ios::out);

// 将读指针移动到文件末尾
file.seekg(0, ios::end);

// 获取当前位置,即文件大小
streampos size = file.tellg();
cout << "文件大小是: " << size << " 字节" << endl;
注意:
  • 在操作完成后,可能需要将文件指针重新定位到适当的位置,以便进行后续的读/写操作。
  • 尝试读取超出文件大小的位置可能会导致未定义的行为,通常是读取失败。

http://www.niftyadmin.cn/n/5084701.html

相关文章

【】javax.crypto.IllegalBlockSizeException: Input length not multiple of 8 bytes

问题描述 jdk版本&#xff1a;8 用DES进行加解密&#xff0c;其中转换模式为“DES/CBC/NoPadding”&#xff0c;要加密的明文为 “密码学浅析”&#xff0c;执行加密操作&#xff0c;报如下错误 Exception in thread "main" javax.crypto.IllegalBlockSizeExcepti…

【Electron+Vue】Error: error:0308010C:digital envelope routines::unsupported

问题描述 使用 electron-builder 构建 vue 项目&#xff0c;运行 npm run electron:build &#xff0c;构建过程报错。 / Bundling main process...ERROR Failed to compile with 1 errors …

Bootstrap-媒体类型

加上媒体查询之后&#xff0c;只有在特定的设备之下才能起作用&#xff01;&#xff01;&#xff01;

【ROS】Nav2源码之nav2_controller详解

1、简述 nav2_controller(Nav2控制器)是Nav2中的任务服务器。根据 nav2_planner 模块计算出路径规划(全局路径或局部路径),生成速度、方向控制的命令(说白了就是控制机器人沿着找到的路走)。 nav2_controller生成速度、方向控制的命令的方法可以有多种,因此它设计成插…

ffmpeg视频解码器的配置选项含义

lowres的含义 lowres是AVCodecContext结构体中的一个成员变量&#xff0c;用于指定编解码器的降低分辨率级别。 在某些情况下&#xff0c;为了加快编解码的速度或减少计算资源的消耗&#xff0c;可以通过设置lowres参数来降低编解码器的分辨率级别。这将导致编解码器在处理视…

YB4618 具有充电前端过电压和过温保护功能,低压差充电前端 OVP 保护开关IC

YB4618 低压差充电前端 OVP 保护开关 概述: YB4618 具有充电前端过电压和过温保护功能。支持 3V 到 40V 的宽输入电压工作范围。过压保护阈值可以外部设置或采用内部默认设置。超快的过压保护响应速度能够确保后级电路的安全。集成了超低导通阻抗的 nFET 开关&#xff0c;确…

C++技能系列( 9 ) - 如何实现线程池【详解】

系列文章目录 C高性能优化编程系列 深入理解软件架构设计系列 高级C并发线程编程 C技能系列 期待你的关注哦&#xff01;&#xff01;&#xff01; 现在的一切都是为将来的梦想编织翅膀&#xff0c;让梦想在现实中展翅高飞。 Now everything is for the future of dream w…

解析项目管理任务跟踪器,助力项目进展掌握!

什么是项目管理任务跟踪器&#xff1f;项目管理任务跟踪器是项目经理简化计划、组织和执行项目任务直至完成的重要工具。该工具可帮助他们掌握需要完成的工作、确定收到的工作请求的优先级、完成项目并在预算范围内按时实现目标。 除了布置和跟踪任务之外&#xff0c;项目管理任…