文件操作

参考 https://fishc.com.cn/forum.php?mod=forumdisplay&fid=329&filter=typeid&typeid=583

打开文件

fopen 打开一个文件并返回文件指针。

#include 
...
FILE *fopen(const char *path, const char *mode);

参数解析:

参数 含义
path 该参数是一个 C 语言字符串,指定了待打开的文件路径和文件名(见备注)
mode 1. 该参数是一个 C 语言字符串,指定了文件的打开模式
2. 下面列举了所有可使用的打开模式:
模式 描述
"r" 1. 以只读的模式打开一个文本文件,从文件头开始读取
2. 该文本文件必须存在
"w" 1. 以只写的模式打开一个文本文件,从文件头开始写入
2.如果文件不存在则创建一个新的文件
3. 如果文件已存在则将文件的长度截断为 0(重新写入的内容将覆盖原有的所有内容)
"a" 1. 以追加的模式打开一个文本文件,从文件末尾追加内容
2. 如果文件不存在则创建一个新的文件
"r+" 1. 以读和写的模式打开一个文本文件,从文件头开始读取和写入
2. 该文件必须存在
3. 该模式不会将文件的长度截断为 0(只覆盖重新写入的内容,原有的内容保留)
"w+" 1. 以读和写的模式打开一个文本文件,从文件头开始读取和写入
2. 如果文件不存在则创建一个新的文件
3. 如果文件已存在则将文件的长度截断为 0(重新写入的内容将覆盖原有的所有内容)
"a+" 1. 以读和追加的模式打开一个文本文件
2. 如果文件不存在则创建一个新的文件
3. 读取是从文件头开始,而写入则是在文件末尾追加
"b" 1. 与上面 6 中模式均可结合("rb", "wb", "ab", "r+b", "w+b", "a+b")
2. 其描述的含义一样,只不过操作的对象是二进制文件(见备注)

返回值:

  1. 如果文件打开成功,则返回一个指向 FILE 结构的文件指针;

  2. 如果文件打开失败,则返回 NULL 并设置 errno 为指定的错误。

备注:

  1. path 参数可以是相对路径(../fishc.txt)也可以是绝对路径(/home/FishC/fishc.txt),如果只给出文件名而不包含路径,则表示该文件在当前文件夹中

  2. 从本质上来说,文本文件也是属于二进制文件的,只不过它存放的是相应的字符编码值。

  3. 打开方式要区分文本模式和二进制模式的原因,主要是因为换行符的问题。C 语言用 \n 表示换行符,Unix 系统用 \n,Windows 系统用 \r\n,Mac 系统则用 \r。如果在 Windows 系统上以文本模式打开一个文件,从文件读到的 \r\n 将会自动转换成 \n,而写入文件则将 \n 替换为 \r\n。但如果以二进制模式打开则不会做这样的转换。Unix 系统的换行符跟 C 语言是一致的,所以不管以文本模式打开还是二进制模式打开,结果都是一样的。

关闭文件

fclose 函数用于关闭先前由 fopen 函数打开的文件。

fclose 函数会将缓冲区内的数据写入文件中,并释放系统所提供的文件资源。

#include 
...
int fclose(FILE *fp);

返回值:

  1. 如果文件关闭成功,返回值是 0;

  2. 如果文件关闭失败,返回值是 EOF,并设置 errno 为指定的错误。

写入文件

  1. fputc
  • 函数用于将一个字符写入到指定的文件中并推进文件的位置指示器(用来指示接下来要读写的下一个字符的位置)。
  • fputc 函数和 putc函数两个的功能和描述基本上是一模一样的,它们的区别主要在于实现上:fputc 是一个函数;而 putc 则是一个宏的实现
#include 
...
int fputc(int c, FILE *stream);
  1. fputs
  • fputs 函数用于将一个字符串写入到指定的文件中,表示字符串结尾的 '\0' 不会被一并写入。
#include 
#include 

int main(void)
{
        FILE *fp;
        int ch;

        if ((fp = fopen("file.txt", "w")) == NULL)
        {
                printf("打开文件失败!\n");
                exit(EXIT_FAILURE);
        }

        fputs("I love FishC.com!\n", fp);

        fclose(fp);

        return 0;
}
  1. fwrite
  • fwrite 函数用于将指定尺寸的数据写入到指定的文件中。
#include 
...
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);
#include 
#include 
#include 

struct Date
{
        int year;
        int month;
        int day;
};

struct Book
{
        char name[40];
        char author[40];
        char publisher[40];
        struct Date date;
};

int main(void)
{
        FILE *fp;
        struct Book *book_for_write, *book_for_read;

        // 为结构体分配堆内存空间
        book_for_write = (struct Book *)malloc(sizeof (struct Book));
        book_for_read = (struct Book *)malloc(sizeof (struct Book));

        if (book_for_write == NULL || book_for_read == NULL)
        {
                printf("内存分配失败!\n");
                exit(EXIT_SUCCESS);
        }

        // 填充结构体数据
        strcpy(book_for_write->name, "《带你学C带你飞》");
        strcpy(book_for_write->author, "小甲鱼");
        strcpy(book_for_write->publisher, "清华大学出版社");
        book_for_write->date.year = 2017;
        book_for_write->date.month = 11;
        book_for_write->date.day = 11;

        if ((fp = fopen("file.txt", "w")) == NULL)
        {
                printf("打开文件失败!\n");
                exit(EXIT_SUCCESS);
        }

        // 将整个结构体写入文件中
        fwrite(book_for_write, sizeof(struct Book), 1, fp);

        // 写入完成,关闭保存文件
        fclose(fp);

        // 重新打开文件,检测是否成功写入数据
        if ((fp = fopen("file.txt", "r")) == NULL)
        {
                printf("打开文件失败!\n");
                exit(EXIT_FAILURE);
        }

        // 在文件中读取结构体并打印到屏幕上
        fread(book_for_read, sizeof(struct Book), 1, fp);

        printf("书名:%s\n", book_for_read->name);
        printf("作者:%s\n", book_for_read->author);
        printf("出版社:%s\n", book_for_read->publisher);
        printf("出版日期:%d-%d-%d\n", book_for_read->date.year, book_for_read->date.month, book_for_read->date.day);

        fclose(fp);

        return 0;
}                           

读取文件

  1. fgetc
  • fgetc 函数用于从文件流中读取下一个字符并推进文件的位置指示器(用来指示接下来要读写的下一个字符的位置)。
  • fgetc 函数和 getc 函数两个的功能和描述基本上是一模一样的,它们的区别主要在于实现上:fgetc 是一个函数;而 getc 则是一个宏的实现
#include 
...
int fgetc(FILE *stream);
  1. fgets
  • fgets 函数用于从指定文件中读取字符串。
  • fgets 函数最多可以读取 size - 1 个字符,因为结尾处会自动添加一个字符串结束符 '\0'。当读取到换行符('\n')或文件结束符(EOF)时,表示结束读取('\n' 会被作为一个合法的字符读取)。
#include 
...
char *fgets(char *s, int size, FILE *stream);
  1. fread
  • fread 函数用于从指定的文件中读取指定尺寸的数据。
#include 
...
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);

缓冲区

fflush 函数用于将缓冲区内的数据强制写入指定的文件中。

#include 
...
int fflush(FILE *stream);

返回值:

  1. 如果函数调用成功,返回值是 0;

  2. 如果函数调用失败,返回 EOF 并设置 errno 为指定的错误。

#include 
#include 

int main(void)
{
        char buff[1024];

        memset(buff, '\0', sizeof(buff));

        // 指定 buff 为缓冲区,_IOFBF 表示当缓冲区已满时才写入 stdout
        setvbuf(stdout, buff, _IOFBF, 1024);

        fprintf(stdout, "This is bbs.fishc.com\n");
        fprintf(stdout, "This output will go into buff\n");

        // fflush强制将上面缓存中的内容写入stdout
        fflush(stdout);

        fprintf(stdout, "this will appear when progream\n");
        fprintf(stdout, "will come after sleeping 5 seconds\n");

        sleep(5);

        return 0;
}

你可能感兴趣的:(文件操作)