文件IO操作

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
char s_readbuf[100];
char *ADP_NET_GetModuleVer_SW()
{
    FILE *fp = NULL;
    int len;
    fp = fopen("./version", "r");
    if (!fp) {
        printf("fopen error\n");
        return 0;
    }
    fseek(fp, 0, SEEK_END);
    len = ftell(fp);
    rewind(fp);
    fread(s_readbuf, 1, len-1, fp);//len-1 去除换行符
    s_readbuf[len] = '\0';
    fclose(fp);
    fp = NULL;
    return s_readbuf;
}
void main()
{
    char *buf = NULL;

    buf = ADP_NET_GetModuleVer_SW();
    printf("file content: %s\n", buf);
}

知识点说明

一、fopen

(1)函数原形

FILE * fopen(const char * path, const char * mode);

(2)参数mode
参数mode说明
(3)二进制和文本模式的区别

  • 在Windows系统中,文本模式下,文件以"\r\n"代表换行。若以文本模式打开文件,并用 fputs 等函数写入换行符"\n"时,函数会自动在"\n"前面加上"\r"。即实际写入文件的是"\r\n"。
  • 在类 Unix/Linux 系统中文本模式下,文件以"\n"代表换行。所以 Linux 系统中在文本模式和二进制模式下并无区别。

(4)返回值

文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回 NULL,并把错误代码存在 error 中。

(5)注意事项

  • 在定义文件指针时,要将文件指针指向空;如 FILE *fp = NULL
  • 在文件操作完成后,别忘记fclose,否则会造成内存泄漏和在下次访问文件时出现问题。
  • 文件关闭后,需要将文件指针指向空,这样做会防止出现游离指针,而对整个工程造成不必要的麻烦;如:fp = NULL

二、fseek

(1)函数原形

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

(2)参数说明

第一个参数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);把stream指针移动到离文件开头100字节处;
fseek(fp,100L,1);把stream指针移动到离文件当前位置100字节处;
fseek(fp,-100L,2);把stream指针退回到离文件结尾100字节处。

(3)返回值

  • 如果执行成功,stream将指向以fromwhere为基准,偏移offset(指针偏移量)个字节的位置,函数返回0。
  • 如果执行失败(比如offset超过文件自身大小),则不改变stream指向的位置,函数返回-1,设置error的值,可以用perror()函数输出错误。

(4)注意事项

  • 文件指针操作文件,会直接覆盖原先的内容。fread fwrite操作都会对文件指针进行偏移。
  • 实现文件内容中插入字符串,先定位到要插入的文件指针位置,将之后的内容保存在缓存中,插入目标字符串后再把文件缓存的内容添加。
  • 函数 ftell 用于得到文件位置指针当前位置相对于文件首的偏移字节数。配合fseek使用。会计算换行符的长度
  • 计算完文件长度后,记得rewind(将文件内部的位置指针重新指向一个流(数据流/文件)的开头)。等价于fseek(stream, 0L, SEEK_SET)。

三、fread

(1)函数原形

size_t fread ( void *buffer, size_t size, size_t count, FILE *stream) ;

(2)参数说明

buffer 用于接收数据的内存地址
size 要读的每个数据项的字节数,单位是字节
count 要读count个数据项,每个数据项size个字节.
stream 输入流

(3)返回值

个人理解为返回数据项数

char *ADP_NET_GetModuleVer_SW()
{
    FILE *fp = NULL;
    int len;
    int ret;

    fp = fopen("./version", "r");
    if (!fp) {
        printf("fopen error\n");
        return 0;
    }
    fseek(fp, 0, SEEK_END);
    len = ftell(fp);
    rewind(fp);
    ret = fread(s_readbuf, 1, len-1, fp);//len-1 去除换行符
    printf("len = %d\n", len);
    printf("fread ret = %d\n", ret);
    //fread(s_readbuf, len-1, 1, fp);
    s_readbuf[len] = '\0';
    fclose(fp);
    fp = NULL;
    return s_readbuf;    
}
void main()
{
    char *buf = NULL;
    int ret;
    buf = ADP_NET_GetModuleVer_SW();
    printf("file content: %s\n", buf);
}

运行结果:

len = 28
fread ret = 27
file content: plt-ec20-0.01
plt-ec20-0.02

当使用 fread(s_readbuf, 1, len, fp),当len小于fp文件中实际的长度,fread的返回值为len,当len大于fp文件中实际的长度。fread的返回值为文件fp实际的长度。

修改为:ret = fread(s_readbuf, 1, 50, fp);
运行结果:
len = 28
fread ret = 28
file content: plt-ec20-0.01
plt-ec20-0.02

修改为:ret = fread(s_readbuf, 1, 5, fp);
运行结果:
len = 28
fread ret = 5
file content: plt-e

当使用 fread(s_readbuf, len, 1, fp),这种语句时,个人理解为读取一个数据项,数据项长度为参数2,当fp文件长度小于len,则说明没有读取完整一个数据项,返回值为0。反之说明已经读取完整一个数据项,返回值为1。

修改为:ret = fread(s_readbuf, 50, 1, fp);
运行结果:
len = 28
fread ret = 0
file content: plt-ec20-0.01
plt-ec20-0.02

修改为:ret = fread(s_readbuf, len, 1, fp);
运行结果:
len = 28
fread ret = 1
file content: plt-ec20-0.01
plt-ec20-0.02

4、fwrite

(1)函数原形

size_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);

(2)参数说明

  • buffer:是一个指针,对fwrite来说,是要获取数据的地址;
  • size:要写入内容的单字节数;
  • count:要进行写入size字节的数据项的个数;
  • stream:目标文件指针;
  • 返回实际写入的数据项个数count。

(3)返回值

返回实际写入的数据块数目

猜你喜欢

转载自blog.csdn.net/chenwr2018/article/details/88060439
今日推荐