• 文件操作


    fopen, _wfopen ,_wfopen_s

    errno_t fopen_s( 
       FILE** pFile,
       const char *filename,
       const char *mode 
    );
    errno_t _wfopen_s(
       FILE** pFile,
       const wchar_t *filename,
       const wchar_t *mode 
    );

    这些是 fopen, _wfopen 的版本与安全增强的 CRT中的安全功能

    Unicode 支持

    fopen_s 支持 Unicode 文件流。 若要打开新的或现有的 Unicode 文件,请通过指定预期编码的 ccs 标志传递给 fopen_s 如下所示:

    fopen_s(&fp, "newfile.txt", "rw, ccs= encoding ");

    encoding 允许的值为 UNICODE、 UTF-8和 UTF-16LE。 如果没有为 encoding指定的值, fopen_s 默认为 ANSI 编码。

    "r"

    用于读取打开。 如果文件不存在或无法找到, fopen_s 调用失败。

    "w"

    打开要编写的空文件。 如果给定文件存在,销毁其内容。

    "a"

    为编写打开隐藏文件 (追加) 结束时,如果不移除 EOF 标记在编写新数据之前到文件中;,如果不存在,则先创建文件。

    "r+"

    用于读取和写入打开。 (文件必须存在。)

    "w+"

    打开读取和写入的空文件。 如果给定文件存在,销毁其内容。

    "a+"

    用于读取和追加打开,追加的操作包括 EOF 标记中删除,新的数据写入文件之前,然后恢复 EOF 标记,在编写完成之后;,如果不存在,则先创建文件。

    当文件打开与 "a" 或 "a+" 访问类型时,所有写入操作发生在文件末尾。 ,在所有写入操作之前,使用 fseek 或 rewind,文件指针可以被重新定位,但是,总是移回文件的结尾。 因此,现有数据无法重写。

    "a" 模式不在追加前移除 EOF 添加到文件。 在追加发生后, MS-DOS 类型命令只显示数据到原始 EOF 标记和没有任何数据追加到文件。 "a+" 模式在追加前移除 EOF 添加到文件。 在追加后, MS-DOS 类型命令文件中显示所有数据。 "a+" 模式对于追加需要于停止点与 CTRL+Z EOF 标记的流文件。

    当 "r+","w+", 或 "a+"访问类型指定,时读取和写入允许 (文件被视为打开 “更新”)。 但是,那么,当您从读取切换到编写,输入操作必须遇到 EOF 标记。 如果没有 EOF,必须使用插入对文件进行标识功能。 文件定位功能是 fsetpos、 fseek和 rewind。 当从文本切换到读取,必须使用干预时调用 fflush 或文件确定的功能。

    除了上面的值,以下字符在 mode 可以包括为换行符指定版本模式:

    t

    打开在文本 (转换) 模式。 在此模式下, CTRL+Z 被解释为编码的文件结尾字符。 如果可能在用于读取打开的文档/与 "a+"的文本, fopen_s 检查 CTRL+Z 在文件末尾并将其移除,。 这样做,因为使用 fseek 和 ftell 在按 CTRL+Z 结束的文件内移动,可能导致 fseek 在文件末尾附近的不正常运行。

    此外,以文本方式,支持返回换行符组合转换为输入的唯一换行符,并且,换行符转换为支持输出中返回换行符组合。 当 Unicode 流 I/O 功能在文本模式 (默认) 下运行时,源或目标流假定为多字节字符序列。 因此, Unicode 流输入函数转换多字节字符转换为宽字符 (如同通过对 mbtowc 函数的调用)。 出于相同的原因, Unicode 流输出函数将宽字符转换为字节字符 (如同通过对 wctomb 函数的调用)。

    b

    打开二进制文件 (未转换的) 模式;涉及回车符和换行符的转换取消。

    如果 t 或 b 在 mode未给出,特定模式的默认值是用全局变量 _fmode定义的。 如果 t 或 b 前缀给参数,则函数将失败并返回 NULL

    有关使用文本和二进制模式的更多信息在 Unicode 和多字节流 I/O,请参见 文本和二进制架构文件 I/O 和 中的 Unicode 文本和二进制模式的流 I/O

    c

    启用关联的 filename 进行标记,以便文件缓冲区的内容直接写入磁盘,如果 fflush 或 _flushall 调用。

    n

    重置关联的 filename 进行标志 “不可更改”。这是默认值。 ,如果使用 COMMODE.OBJ,链接程序它还重写全局进行标记。 全局标志使默认值为 “无使”,除非您使用 COMMODE.OBJ 显式链接程序 (请参见 LINK选项)。

    N

    指定文件未由子进程继承。

    S

    指定缓存中优化对,但是,不受限制为,顺序访问从磁盘。

    R

    指定缓存中优化对,但是,不受限制为,随机访问从磁盘。

    T

    指定文件。临时。 如果可能,它不会刷新到磁盘。

    D

    指定文件。临时。 ,在最后一个文件指针关闭时,则会将其删除。

    ccs=ENCODING

    指定该代码的字符集为此文件使用 (UTF-8、 UTF-16LE 和 UNICODE)。 

    int fseek(FILE *stream, long offset, int fromwhere);

    fseek 用于二进制方式打开的文件,移动文件读写指针位置.
    fseek(in,-1L,1);   -- 文件流in, 零点为当前指针位置,SEEK_CUR 就是 1,  -1L -- 文件指针回退1个字节int fseek( FILE *stream, long offset, int origin );
    第一个参数stream为文件指针
    第二个参数offset为偏移量,整数表示正向偏移,负数表示负向偏移
    第三个参数origin设定从文件的哪里开始偏移,可能取值为:SEEK_CUR、 SEEK_END 或 SEEK_SET
    SEEK_SET: 文件开头
    SEEK_CUR: 当前位置
      SEEK_END: 文件结尾
      其中SEEK_SET,SEEK_CUR和SEEK_END和依次为0,1和2.
      简言之:
      fseek(fp,100L,0);把fp指针移动到离文件开头100字节处;
      fseek(fp,100L,1);把fp指针移动到离文件当前位置100字节处;
        fseek(fp,100L,2);把fp指针退回到离文件结尾100字节处。

    filesize = ftell(fp);获取文件大小

    fread函数和fwrite函数

     

    1.函数功能

      用来读写一个数据块。

    2.一般调用形式

      fread(buffer,size,count,fp);

      fwrite(buffer,size,count,fp);

    3.说明

      (1)buffer:是一个指针,对fread来说,它是读入数据的存放地址。对fwrite来说,是要输出数据的地址。

      (2)size:要读写的字节数;

      (3)count:要进行读写多少个size字节的数据项;

      (4)fp:文件型指针。

     注意:1 完成次写操(fwrite())作后必须关闭流(fclose());

               2 完成一次读操作(fread())后,如果没有关闭流(fclose()),则指针(FILE * fp)自动向后移动前一次读写的长度,不关闭流继续下一次读操作则接着上次的输出继续输出;

               3 fprintf() : 按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了。注意的是返回值为此次操作写入到文件的字节数。如int c = fprintf(fp, "%s %s %d %f", str1,str2, a, b) ;str1:10字节;str2: 10字节;a:2字节;b:8字节,c为33,因为写入时不同的数据间自动加入一个空格。

    文件使用之后一定要关闭,否则将不能正确显示内容.fwrite:读入两个学生信息然后用fwrite存入文件

    fread:用fread从文件中读出学生信息。

    fwrite.c

    #include <stdio.h>
    #define SIZE 2
    struct student_type
    {
     char name[10];
     int num;
     int age;
     char addr[10];
    }stud[SIZE];
    void save()
    {
     FILE *fp;
     int i;
     if((fp=fopen("stu_list","wb"))==NULL)
     {
      printf("cant open the file");
      exit(0);
     }
     for(i=0;i<SIZE;i++)
     {
       if(fwrite(&stud[i],sizeof(struct student_type),1,fp)!=1)
        printf("file write error\n");
     }
     fclose(fp);
    }
    main()
    {
     int i;
     for(i=0;i<SIZE;i++)
     {
       scanf("%s%d%d%s",&stud[i].name,&stud[i].num,&stud[i].age,&stud[i].addr);
       save();
     }
     for(i=0;i<SIZE;i++)
     {
       printf("%s,%d,%d",stud[i].name,stud[i].num,stud[i].age,stud[i].addr);
     }
    }

    fread.c

    #include <stdio.h>
    #define SIZE 2
    struct student_type
    {
     char name[10];
     int num;
     int age;
     char addr[10];
    }stud[SIZE];
    void read()
    {
     FILE *fp;
     int i;
     if((fp=fopen("stu_list","rb"))==NULL)
     {
      printf("cant open the file");
      exit(0);
     }
     for(i=0;i<SIZE;i++)
     {
       if(fread(&stud[i],sizeof(struct student_type),1,fp)!=1)
        printf("file write error\n");
     }
     fclose(fp);
    }
    main()
    {

     int i;
     read();
     for(i=0;i<SIZE;i++)
     {
       printf("%s,%d,%d,%s",stud[i].name,stud[i].num,stud[i].age,stud[i].addr);
       printf("\n");
     }
    }

    在C语言中进行文件操作时,我们经常用到fread()和fwrite(),用它们来对文件进行读写操作。下面详细绍一下这两个函数的用法。

      我们在用C语言编写程序时,一般使用标准文件系统,即缓冲文件系统。系统在内存中为每个正在读写的文件开辟“文件缓冲区”,在对文件进行读写时数据都经过缓冲区。要对文件进行读写,系统首先开辟一块内存区来保存文件信息,保存这些信息用的是一个结构体,将这个结构体typedef为FILE类型。我们首先要定义一个指向这个结构体的指针,当程序打开一个文件时,我们获得指向FILE结构的指针,通过这个指针,我们就可以对文件进行操作。例如:

    #i nclude <stdio.h>

    #i nclude <string.h>

    int main()

    {

       FILE *fp;

       char buffer[100] = "This is a test";

       if((fp = fopen("c:\\example.txt", "w")) == 0)

        {

           printf("open failed!");

           exit(1);

        }

       fwrite(buffer, 1, strlen("This is a test"), fp);

       fclose(fp);

       return 0;

    }

      通过以上代码,我们就在c盘的根目录下建立了一个名为example扩展名为.txt的文件,我们打开可以看到上面写上了This is a test。当我们对它将它读出时,用如下代码:

    #i nclude <stdio.h>

    #i nclude <mem.h>

    int main()

    {

       FILE *fp;   int len;

       char buffer[100];

       /*memset(buffer, 1, 100); */

       if((fp = fopen("c:\\example.txt", "r")) == 0)

        {

           printf("open failed!");

           exit(1);

        }

       fseek(fp, 0L, SEEK_END);

       len = ftell(fp);

       rewind(fp);

       fread(buffer, 1, len , fp);

       printf("%s",buffer);

       fclose(fp);

       getch();

       return 0;

    }

     可以看到,当我们使用memset了以后,读出了一大堆乱码,这是为什么呢?原因是我们在fwrite函数时写入的字节数是用strlen求得的,也就是说字符串最后的'\0'并没有写到文件中去。所以我们从文件中读到buffer中时也自然没有'\0',因为buffer中的数是随机的,除非buffer中最后一个字符的下一个数恰好随机到0(可能性很小,这里用memset将它排除),否则以%s将buffer中的字符输出时遇不到0,所以乱码产生。解决的办法有很多,你可以在向文件写数据时多写入一个字节,系统会自动写入0,fwrite(buffer, 1, strlen("This is a test")+1, fp);这样读出时最后就有一个0了。或者读出操作完成后,在最后一个字符后面补上一个0:buffer[len] = 0;这样问题也可得到解决。

  • 相关阅读:
    PO BO VO DTO POJO DAO DO这些Java中的概念分别指一些什么?
    前端面试题汇总(待续)
    vue lottie vue-lottie : 使用教程
    webstorm 换行时 代码不对齐
    webstorm 导出编辑器配置.editorconfig
    vue 查看dist文件里的结构
    vue-cli 生产打包
    element form 校验数组每一项
    typescript无法识别vue中的$refs
    mac 10.14.5 [vue create的时候 mkdir没有权限]
  • 原文地址:https://www.cnblogs.com/fripside/p/2935157.html
Copyright © 2020-2023  润新知