C语言中文件常见操作的示例详解

目录
  • 文件打开和关闭
  • 文件写入
  • 文件读取
  • fseek函数
  • ftell函数
  • Demo示例
  • 解决读取乱码

FILE为C语言提供的文件类型,它是一个结构体类型,用于存放文件的相关信息。文件打开成功时,对它作了内存分配和初始化。

每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。

一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。

文件打开和关闭

C语言的安全文件打开函数为_wfopen_s和_fopen_s。

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

参数pFile

指向文件指针的指针,该文件指针将接收指向打开文件的指针。

参数filename

表示要打开的文件名

参数mode

表示访问方式,该参数可设置的值如下表。

模式 含义
"r" 打开以供阅读。如果文件不存在或找不到,则fopen_s调用将失败
"w" 打开一个空文件进行写入。如果给定文件存在,则其内容将被销毁
"a" 在新数据写入文件之前,在文件末尾打开以写入(追加),而不删除文件结束 (EOF) 标记。如果文件不存在,则创建该文件
"r+" 可打开读取和写入。该文件必须存在
"w+" 打开一个空文件进行读取和写入。如果该文件存在,则其内容将被销毁
"a+"  打开以供阅读和追加。追加操作包括在将新数据写入文件之前删除 EOF 标记。写入完成后,不会还原 EOF 标记。如果文件不存在,则创建该文件

除了上述模式外,还可以包含以下字符,以指定换行符的转换模式:

模式修改器 翻译模式
t 以文本(翻译)模式打开。
b 以二进制(未翻译)模式打开;禁止显示涉及回车符和换行符的转换。

返回值 

如果成功,则为零;失败时的错误代码

使用Close函数文件关闭。

文件写入

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

参数buffer

表示存储读取数据的缓冲区

参数size

表示要写入数据的一个字符的大小(以字节为单位)

参数count

表示要写入数据的字符数

参数stream

表示指向文件结构体的指针

返回值

fwrite返回函数写入的完整项数,如果发生错误,该数可能小于count。

文件读取

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

参数buffer

表示存储读取数据的缓冲区

参数size

表示要读取数据的一个字符的大小(以字节为单位)

参数count

表示要读取数据的字符数

参数stream

表示指向文件结构体的指针

返回值

fread返回函数读取的完整项目数,如果发生错误,或者在达到COUNT 之前遇到文件末尾,则可能小于Count。

fseek函数

可以通过fseek函数设置文件指针的位置。通过该方法可以计算出文件的大小

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

参数stream

表示指向文件结构体的指针

参数offset

表示指针的偏移量

参数origin

表示指针所处位置 ,其可以设置的如下的取值

取值 意义
SEEK_CUR 文件指针的当前位置
SEEK_END 文件结束
SEEK_SET 文件的开头

ftell函数

ftell函数可以获取文件指针的当前位置。通过fseek结合ftell两个函数可以计算出文件的大小

long ftell(
   FILE *stream
);

Demo示例

写数据:

void CMyFileCFileView::OnFileWrite() {
    FILE* pFile = NULL;
    //打开文件
    errno_t err = _wfopen_s(&pFile, _T("1.txt"), _T("w"));
    if (err != 0) {
        TRACE("Open File failed errorcode :%d", GetLastError());
        return;
    }
    fwrite(L"Hello World", 2, wcslen(L"Hello World")+1, pFile);
    fclose(pFile);
}

读数据:

void CMyFileCFileView::OnFileRead() {
    FILE* pFile = NULL;
    errno_t err = _wfopen_s(&pFile, _T("1.txt"), _T("r"));
    if (err != 0) {
        TRACE("Open File failed errorcode :%d", GetLastError());
        return;
    }
    fseek(pFile, 0, SEEK_END);//文件指针定位,偏移到结尾
    int len = ftell(pFile);//得到文件指针的当前位置 = 每个字符大小×字符长度
    WCHAR* pBuf = new WCHAR[len];
    fseek(pFile, 0, SEEK_SET);//文件指针回到文件起始位置
    pBuf[len] = 0;
    fread(pBuf, 2, len, pFile);
    MessageBox(pBuf);
    fclose(pFile);
}

解决读取乱码

使用C语言进行文件操作时如果稍微不注意很容易读出乱码。一般出现这种乱码有两种原因,一种时写入的数据没有写入结束符,另一种是读取数据时,读取数据的个数设置出错。

例如下面这种写入数据和读取数据的代码,运行出来就会出现乱码。

//写数据
fwrite(L"Hello World", 2, wcslen(L"Hello World"), pFile);

//读数据
//pFile此时移动到了文件末尾
int len = ftell(pFile);//得到文件指针的当前位置 = 每个字符大小×字符长度
WCHAR* pBuf = new WCHAR[len];
fseek(pFile, 0, SEEK_SET);//文件指针回到文件起始位置
fread(pBuf, 2, len, pFile);

使用ftell获取的文件指针到的位置为 一个字符大小 ×字符长度。 虽然"Hello World"有11个字符,但是WCHAR类型,所以ftell返回的值为22。在之后使用fread读取数据时读到了len×2=44字节的内容,而写入的数据才22字节,因此会出现乱码。

解决方法:

方法一:给存储的数据结尾加上结束符'\0"

在写数据时:

fwrite(L"Hello World", 2, wcslen(L"Hello World")+1, pFile);

写入的实际长度+1,这样写入的数据后面会增加一个'\0'结束符。

fread在达到COUNT 之前遇到文件末尾也会停止。这样在缓冲缓冲数组就可以读到实际数据和一个结束符。打印时就能正常打印出。

//写数据
fwrite(L"Hello World", 2, wcslen(L"Hello World")+1, pFile);

//读数据
//pFile此时移动到了文件末尾
int len = ftell(pFile);//得到文件指针的当前位置 = 每个字符大小×字符长度
WCHAR* pBuf = new WCHAR[len];
fseek(pFile, 0, SEEK_SET);//文件指针回到文件起始位置
fread(pBuf, 2, len, pFile);

方法二:限制读取的大小

因为len得到的是一个字符的大小×字符数,如果把len/sizeof(WCHAR),就可以获得实际的字符数。在读取数据时使用这个实际的长度作为读取数,之后将缓冲数组len的位置赋值为'\0'。就可以读取到实际数据

//写数据
fwrite(L"Hello World", 2, wcslen(L"Hello World"), pFile);

//读数据
//pFile此时移动到了文件末尾
int len = ftell(pFile);//得到文件指针的当前位置 = 每个字符大小×字符长度
WCHAR* pBuf = new WCHAR[len];
fseek(pFile, 0, SEEK_SET);//文件指针回到文件起始位置
fread(pBuf, 2, len/2, pFile);
pBuf[len/2] = 0;

到此这篇关于C语言中文件常见操作的示例详解的文章就介绍到这了,更多相关C语言文件操作内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言文件操作详解

    目录 一.什么是文件 二.文件缓冲区 三.文件指针 四.文件的打开和关闭. 总结 一.什么是文件 在程序设计中,我们一般谈的文件有两种:程序文件.数据文件. 程序文件: 包括源程序文件(后缀为.c ),目标文件( windows环境后缀为.obj ) ,可执行程序( windows环境后缀为.exe ). 数据文件: 文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件 或者输出内容的文件. 数据文件又分为"文本文件"和"二级制文件"

  • C语言全面细致讲解文件操作

    目录 什么是文件 程序文件 数据文件 文件名 文件指针 文件的打开和关闭 文件的顺序读写 字符输入输出函数 字符串输入输出函数(fgets,fputs) 格式化输入输出函数(fscanf,fprintf) 二进制输入输出函数(fread,fwrite) 文件的随机读写 fseek ftell 文件读取结束的判定 什么是文件 磁盘上的文件是文件.但是在程序设计中,我们一般谈的文件有两种:程序文件.数据文件(从文件功能的角度来分类的). 程序文件 包括源程序文件(后缀为.c),目标文件(window

  • 一文搞懂C语言中的文件操作

    目录 一.文件操作 1. 为什要使用文件操作 2.什么是文件 3.文件操作的使用 一.文件操作 1. 为什要使用文件操作 在c语言中我们完成一个程序后,他并不会对我们的数据进行保存,就像我上一篇文章中实现一个通讯录人员的增删插改中数据只是短暂的保存,而当程序退出后的同时人员信息也就销毁了,在此执行程序的时候人员信息并没有保存,而这是为什呢?是因为我们执行程序时我们写的代码信息数据不是持久化的,而要让他持久化就要用到文件操作,将我们写的数据存到文件中,而文件信息是存放到硬盘当中的,这样就达到了信息

  • C语言文件操作之fread函数详解

    目录 前言 一.fread 函数 二.缓冲区受限的情况 ( 循环读取文件 | feof 函数判定文件读取完毕 ) 三.处理乱码问题 四.记录读取的字节个数 五.读取到 0 字节的情况 六.读取完毕的情况 七.读取文本文件 “\n” 与 读取二进制文件 “\r\n” 区别 总结 前言 二进制文件读写两个重要的函数 , fread 和 fwrite , fread 用于读取文件 , fwrite 用于写出文件 ; fread / fwrite 函数 既可以操作 二进制文件 , 又可以操作 文本文件

  • C语言超详细讲解文件的操作

    目录 一.为什么使用文件 二.什么是文件 1.程序文件 2.数据文件 3.文件名 三.文件指针 四.文件的打开和关闭 五.文件的顺序读写 六.文件的随机读写 fseek ftell rewind 七.文件结束判定 一.为什么使用文件 当我们写一些项目的时候,我们应该要把写的数据存储起来.只有我们自己选择删除数据的时候,数据才不复存在.这就涉及到了数据的持久化的问题,为我们一般数据持久化的方法有,把数据存在磁盘文件.存放到数据库等方式.使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久

  • C语言常见的文件操作函数

    目录 一.文件的打开和关闭 1.文件指针 2.文件打开和关闭 二.文件的顺序读写 1.fgetc()和fputc()函数 2.fgets()和fputs()函数 3.fscanf()和fprintf()函数 4.fread()和fwrite()函数 三.文件的随机读写 1.fseek函数 2.ftell函数 3.rewind函数 四.文本文件和二进制文件 五.文件读取结束的判定 六.文件缓冲区 总结 一.文件的打开和关闭 1.文件指针 每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存

  • C语言中文件常见操作的示例详解

    目录 文件打开和关闭 文件写入 文件读取 fseek函数 ftell函数 Demo示例 解决读取乱码 FILE为C语言提供的文件类型,它是一个结构体类型,用于存放文件的相关信息.文件打开成功时,对它作了内存分配和初始化. 每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节. 一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便. 文件打开和关闭 C语言的安全文件打开函数为_wfopen_s和_fopen_s

  • Go语言中的字符串处理方法示例详解

    1 概述 字符串,string,一串固定长度的字符连接起来的字符集合.Go语言的字符串是使用UTF-8编码的.UTF-8是Unicode的实现方式之一. Go语言原生支持字符串.使用双引号("")或反引号(``)定义. 双引号:"", 用于单行字符串. 反引号:``,用于定义多行字符串,内部会原样解析. 示例: // 单行 "心有猛虎,细嗅蔷薇" // 多行 ` 大风歌 大风起兮云飞扬. 威加海内兮归故乡. 安得猛士兮守四方! ` 字符串支持转义

  • Python中字典常用操作的示例详解

    目录 前言 初始化 合并字典 字典推导式 Collections 标准库 字典转 JSON 字典转 Pandas 前言 字典是Python必用且常用的数据结构,本文梳理常用的字典操作,看这个就够了,涉及: 初始化 合并字典 字典推导式 Collections 标准库 字典转JSON 字典转Pandas 初始化 # 最常用这种 my_object = { "a": 5, "b": 6 } # 如果你不喜欢写大括号和双引号: my_object = dict(a=5,

  • Go语言中循环语句使用的示例详解

    目录 一.概述 1. 循环控制语句 2. 无限循环 二.Go 语言 for 循环 1. 语法 2. for语句执行过程 3. 示例 4. For-each range 循环 三.循环嵌套 1. 语法 2. 示例 四.break 语句 1. 语法 2. 示例 五. continue 语句 1. 语法 2. 示例 六.goto 语句 1. 语法 2. 示例 一.概述 在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句. 循环程序的流程图: Go 语言提供了以下几种类型循环

  • Go语言中的延迟函数defer示例详解

    前言 大家都知道go语言的defer功能很强大,对于资源管理非常方便,但是如果没用好,也会有陷阱哦.Go 语言中延迟函数 defer 充当着 try...catch 的重任,使用起来也非常简便,然而在实际应用中,很多 gopher 并没有真正搞明白 defer.return.返回值.panic 之间的执行顺序,从而掉进坑中,今天我们就来揭开它的神秘面纱!话不多说了,来一起看看详细的介绍吧. 先来运行下面两段代码: A. 匿名返回值的情况 package main import ( "fmt&qu

  • Python常见文件操作的示例详解

    目录 从文件中读取数据 为什么要提供文件路径 逐行读取 创建一个包含文件各行内容的列表 使用文件中的内容 包含千位以上的大型文件 圆周率中包含你的生日吗 写入文件 附加到文件 从文件中读取数据 1:读取整个文件 首先创建一个文件,它包含一些文本信息,注意:如果该文件为.py文件,则再进行操作的时候可以不写路径,如果不是.py文件,则必须标明路径. 现在我们创建一个python文件,名为text.py,给他写入如下文本信息: 下面的程序打开并读取这个文件,再将其内容显示到屏幕上: with ope

  • C语言编程C++旋转字符操作串示例详解

    目录 旋转字符串 字符串左旋 题前认知: 暴力移位: 三步翻转: 判断字符串旋转 题前认知 字符串追加判断 旋转字符串 字符串左旋 实现一个函数,可以左旋字符串中的k个字符. 例如: ABCD左旋一个字符得到BCDA ABCD左旋两个字符得到CDAB 题前认知: 一个字符串如果就定死了.eg:char arr[]="dfdf"什么的那多没意思,一点都没有人机交互的感觉,(虽然现在人机交互适合个体,不适合集群,但也是比死板的定死字符串舒服) 所以字符串得是我们可输入的,才有可玩性,玩的不

  • C语言中关于动态内存分配的详解

    目录 一.malloc 与free函数 二.calloc 三.realloc 四.常见的动态内存的错误 [C语言]动态内存分配 本期,我们将讲解malloc.calloc.realloc以及free函数. 这是个动态内存分配函数的头文件都是 <stdlib.h>. c语言中动态分配内存的函数,可能有些初学c语言的人不免要问了:我们为什么要通过函数来实现动态分配内存呢? 首先让我们熟悉一下计算机的内存吧!在计算机的系统中大致有这四个内存区域: 1)栈:在栈里面储存一些我们定义的局部变量以及形参(

  • Go语言学习教程之反射的示例详解

    目录 介绍 反射的规律 1. 从接口值到反射对象的反射 2. 从反射对象到接口值的反射 3. 要修改反射对象,该值一定是可设置的 介绍 reflect包实现运行时反射,允许一个程序操作任何类型的对象.典型的使用是:取静态类型interface{}的值,通过调用TypeOf获取它的动态类型信息,调用ValueOf会返回一个表示运行时数据的一个值.本文通过记录对reflect包的简单使用,来对反射有一定的了解.本文使用的Go版本: $ go version go version go1.18 dar

  • Java 语言守护线程 Daemon Thread使用示例详解

    目录 守护线程 用户线程设为守护线程 守护线程 在Java语言中有两类线程:用户线程和守护线程.我们通俗的讲,任意一个守护线程都是整个JVM中所有线程的"大管家":只要当前Java虚拟机中还有任意一个非守护线程没有结束,它们的守护线程就不能结束,要持续工作:只有当最后一个非守护线程结束时,守护线程才随着Java虚拟机一起结束工作. 其作用就是为其他线程的运行提供服务,就像是一个护道者,保证其他线程的顺利运行 用户线程设为守护线程 我们将用户线程设为守护线程的办法就是Thread类的se

随机推荐