C/C++文件IO函数

文件(file)通常是在磁盘或固态硬盘上的一段已命名的存储区。C中采用的主要是文件指针的办法,C++中对文件的操作主要运用了“文件流”(即非标准的输入输出)的思想。

open是linux下的底层系统调用函数,fopen与freopen是c/c++的标准I/O库函数,带输入/输出缓冲。linxu下的fopen是open的封装函数,fopen最终还是要调用底层的系统调用open。所以在linux下如果需要对设备进行明确的控制,使用底层系统调用(open)

open对应的文件操作有:close, read, write,ioctl 等。

fopen 对应的文件操作有:fclose, fread, fwrite, freopen, fseek, ftell, rewind等。

open和fopen的区别:

1,fread是带缓冲的,read不带缓冲.

2,fopen是标准c里定义的,open是POSIX中定义的.  

3,fread和fwrite参数为文件指针,read和write为文件描述符

4,fopen不能指定要创建文件的权限.open可以指定权限.

5,fopen返回文件指针,open返回文件描述符(整数). 

6,linux/unix中任何设备都是文件,都可以用open,read.

一、C

C把文件看作是一系列连续的字节,每个字节都能被单独读取。C提供两种文件模式:文本模式和二进制模式。

1.fopen,fdopen

std::FILE* fopen( const char* filename, const char* mode );
FILE *fdopen(int fd, const char* mode);

1、fopen函数:打开文件filename并返回与该文件关联的文件流。mode用于确定文件访问模式。

2、fdopen函数:打开已存在的文件描述符,使标准I/O流与该文件相结合。主要用于fopen不能打开的特殊文件(如管道和网路通信等),这时必须先调用设备专用函数获得一个文件描述符,然后用fdopen使一个标准I/O与该文件描述符相结合。

参数

filename-将文件流关联到的文件名
mode   -以空字符结尾的字符串确定文件访问模式

返回值

如果成功,则返回一个指向控制已打开文件流的对象的指针,同时清除 eof 和错误位。出错时,返回一个空指针。

2.fclose

int fclose(FILE *stream)

C 库函数 int fclose(FILE *stream) 关闭流 stream。刷新所有的缓冲区。

参数

stream 是指向 FILE 对象的指针,该 FILE 对象指定了要被关闭的流。

返回值

如果流成功关闭,则该方法返回零。如果失败,则返回 EOF。

3.FILE结构体

C语言的stdio.h头文件中定义了用于文件操作的结构体FILE。通过fopen返回一个文件指针(指向FILE结构体的指针)来进行文件操作。可以在stdio.h头文件中查看FILE结构体的定义,如下:

 不过在visual studio中貌似被隐藏了,见https://docs.microsoft.com/en-us/cpp/porting/visual-cpp-change-history-2003-2015

4.fprintf()fscanf()函数

文件I/O函数 fprintf() fscanf() 函数的工作方式与 printf() scanf()类 似,区别在于前者需要用第1个参数指定待处理的文件。
int fprintf(FILE *stream, const char *format, ... );
fprintf()函数根据指定的format(格式)发送信息(参数)到由stream(流)指定的文件.

int fscanf(FILE *stream, const char *format, ...)
C库函数 int fscanf(FILE *stream, const char *format, ...) 从流stream读取格式化输入。
  • stream -- 指向 FILE 对象的指针,该 FILE 对象标识了流。
  • format -- C 字符串,包含了以下各项中的一个或多个:空格字符、非空格字符 和 format 说明符

成功则返回成功赋值或写入的个数,失败或到达文件末尾返回负数。

5.fread和fwrite

这两个函数对应fopen.

fread()
头文件:#include<stdio.h>

size_t fread(void*buffer, size_t size, size_t count, FILE*stream);
1.buffer:  读取的数据存放的内存的指针,(可以是数组,也可以是新开辟的空间)
指向缓冲区保存或读取的数据或者是用于接收数据的内存地址
2.size:   每次读取的字节数
3.count:  读取的次数
4.stream:  是要读取的文件的指针,是数据读取的流(输入流)
返回值:

成功:实际读取的元素(并非字节)数目;失败:返回0
如果输入过程中遇到了文件尾或者出错,可能比请求的元素数目小

fwrite()
头文件:#include<stdio.h>

size_t fwrite(void*buffer, size_ size, size_t count, FILE*stream)
1.buffer:指向用于保存数据的内存位置的指针。对于fwrite来说,是要获取数据的地址
2.size:   每次读取的字节数
3.count:  读取的次数
4.stream: 数据写入的流(目标指针的文件)
返回值:实际写入的元素(并非字节)数目
如果输入过程中遇到了文件尾或者出错,可能比请求的元素数目小

二、C++

头文件fstream 定义了三个类型来支持文件IO: ifstream从一个给定文件读取数据,ofstream向一个给定文件写入数据,以及fstream可以读写给定文件。

每个流都有一个关联的文件模式(file mode),用来指出如何使用文件。表8.4列出了文件模式和它们的含义。

每个文件流类型都定义了一个默认的文件模式,当未指定文件模式时就使用此默认模式。与ifstream关联的文件默认以in模式打开,与ofstream关联的文件默认以out模式打开,与fstream关联的文件默认以in和out模式打开。

默认情况下,打开一个ofstream时,文件的内容会被丢弃。阻止一个ofstream清空给定文件内容的方法是同时指定app或in模式:

//在这几条语句中,filel都被截断
ofstream out ("filel"); //隐含以输出模式打开文件并截断文件
ofstream out2("file1", ofstream::out); //隐含地截断文件
ofstream out3("file1", ofstream::out | ofstream::trunc);
//为了保留文件内容,显式指定app模式
ofstream app("file2", ofstream::app); //隐含为输出模式
ofstream app2("fi1e2", ofstream::out | ofstream::app);

三、示例程序

从一个文件中读数据到另一个文件:

#include<iostream>
using namespace std;

int main()
{
    FILE* fdRead = fopen("read.txt", "r");
    FILE* fdwrite = fopen("write.txt", "a+");
    char buffer[4];
    int n = 0;
    while (n = fread(buffer, sizeof(buffer), 1, fdRead))
    {
        cout <<"n = " << n << ' ';
        int m = fwrite(buffer, sizeof(buffer), 1, fdwrite);
        cout << "m = " << m << ' ';
        cout << endl;
    }
    fclose(fdRead);
    fclose(fdwrite);
    return 0;
}

 C/C++混合写法:

#include<iostream>
#include<string>
#include<algorithm>
#include<sstream>
#include<fstream>
#include<stdio.h>
using namespace std;

const int MAX_NUM = 100;
int a[MAX_NUM];
int n = 2;

int main(int argc, char* argv[])
{
    FILE* file4 = fopen("d.txt", "w");
    if (!file4) {
        printf("file4 open error!\n");
        return -1;
    }
    //fprintf(file4, "name age sex position\n");
    int age, sex;
    char name[50], position[50];
    printf("please input 2 date : name age sex position:\n");
    while (n--)
    {
        scanf("%s %d %d %s", &name, &age, &sex, &position);
        fprintf(file4, "%s %d %d %s\n", name, age, sex, position);
    }
    fclose(file4);
    FILE* file5 = fopen("d.txt", "r");
    if (!file5) {
        printf("file5 open error!");
        return -1;
    }
    int m = 2, ans, ans1;
    while (m--)
    {
        fscanf(file5, "%s %d %d %s", &name, &age, &sex, &position);
        printf("%s %d %d %s\n", name, age, sex, position);
    }
    fclose(file5);

    fstream file1("d.txt");
    if (!file1) {
        cout << "file1 open error! " << endl;
        return -1;
    }
    string tmp;
    vector<string>str;
    while (file1 >> tmp)
    {
        str.push_back(tmp);
        cout << tmp << endl;
    }
    ifstream file2("b.txt", ios::in);
    if (!file2) {
        cout << "file2 open error! " << endl;
        return -1;
    }
    ofstream file3("c.txt", ios::out);
    if (!file3) {
        cout << "file3 open error! " << endl;
        return -1;
    }
    while (n--)
    {
        file3 << n<<' ';
    }
    file1.close();
    file2.close();
    file3.close();
    return 0;
}

输入输出结果如下:

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值