草庐IT

写程序必会的C语言文件操作(上)附手绘图详解

陈大大陈 2023-09-06 原文

目录

1. 为什么使用文件

 2. 什么是文件

2.1 程序文件 

2.2 数据文件

2.3 文件名

3. 文件的打开和关闭

3.1 文件指针

3.2 文件的打开和关闭实例

4. 文件的顺序读写

字符输入函数 fgetc

文本行输出函数 fputs

文本行输入函数 fgets 

格式化输出函数 fprintf

格式化输入函数 fscanf

sprintf 

sscanf

将字符串转换成格式化的数据


1. 为什么使用文件

  • 在之前我们写过一个通讯录的博客,在通讯录里我们可以随意增删查改人员的信息,可是此时数据的信息是存放到内存中的,一旦程序退出,数据也将不复存在。下次使用通讯录时,就需要重新录入数据。
  • 我们希望输入的数据可以一直存在,除非我们主动选择删除。这就涉及到了数据的持久化问题。
  • 我们一般进行数据持久化的方式有将数据存放到磁盘,存放到数据库等。
  • 而我们可以直接将数据存储文件里,再将文件存到电脑的硬盘上,就可以做到数据的持久化。

 2. 什么是文件

磁盘上的文件就是文件。

然而在程序设计中,我们所谈的文件有两种,一种是程序文件,另一种是数据文件(从文件功能的角度来分类的)。

2.1 程序文件 

程序文件又包含源程序文件(后缀为.c),目标文件(Windows环境后缀为.obj),可执行程序(Windows环境后缀为.exe)。

2.2 数据文件

程序文件的数据信息需要存放到数据文件中,我们既可以从程序文件往数据文件里写(输出)信息,也可以从数据文件中向程序文件往外读(输入)信息。

咱们之前所处理数据的输入输出都是以终端为对象的,即从终端的键盘读入数据,运行结果到显示器上,数据是存放到内存里面的。

 作了图方便大家理解。

其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理 的就是磁盘上文件。

那我们应该怎么把内存中的东西写入文件中呢?

2.3 文件名

一个文件要有一个独有的文件标识,便于用户识别和引用。

文件标识包含文件路径,文件名主干和文件后缀。

例如: f:\class\test.txt

其中 f:\class\是文件路径,test是文件名主干,.txt是文件后缀。

为了方便起见,文件标识常被称为文件名

3. 文件的打开和关闭

3.1 文件指针

缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”

 每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息。

 这个文件信息区究竟是什么类型呢?让我们在VS编译器中stdio.h文件中搜寻一番。

可以看到,这些信息是保存在一个结构体变量中的。该结构体类型是由系统 声明的,取名FILE。

不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。 

我们一般可以通过一个FILE指针来维护FILE结构的变量。

下面我们创建一个FILE*的指针变量。

3.2 文件的打开和关闭实例

我们在试着以只读的方式在桌面上打开一个叫“你快乐吗”的文件。

如果文件不存在,打印错误,并返回一个空指针。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fclose(pf);
	pf == NULL;
	return 0;
}

 

 如图,提示没有这个文件。

我们这次以只写“w”的方式打开文件。只写方式如果指定文件不存在,建立一个新的文件。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fclose(pf);
	pf == NULL;
	return 0;
}

程序没有报错。

 

回到桌面,我们创建了一个名叫“你快乐吗”的文件,这个文件本来是没有的。

4. 文件的顺序读写

字符输入函数 fgetc

 我们使用字符输入函数fgetc向上面的文件里写入26个英文字母。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int i = 0;
	for (i = 0; i < 26; i++)
	{
		fputc('a'+i, pf);
	}
	fclose(pf);
	pf == NULL;
	return 0;
}

上面程序运行完之后,文件里出现26个英文字母。

int fgetc ( FILE * stream );

函数的返回类型是int,即返回一个ASCII值,读取错误返回一个EOF。

我们也可以从文件里读出数据。

改成只读的形式后用fgetc读取之后打印出来,我们可以这样写。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int i = 0;
	int ch = 0;
	for (i = 0; i < 26; i++)
	{
		ch=fgetc(pf);
		printf("%c ", ch);
	}
	fclose(pf);
	pf == NULL;
	return 0;
}

运行结果: 

 可以看出,fgetc函数读完一个值后,会让指针向后移一位。

文本行输出函数 fputs

 用fputs可以操作字符串内容。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("hello world", pf);
	fclose(pf);
	pf == NULL;
	return 0;
}

 

文本行输入函数 fgets 

 用fgets读取数据到内存。

char * fgets ( char * str, int num, FILE * stream );

实际上它所读取到的数据比我们所输入的数据少一个,比如我们输入num为5,那么它只能读4个数据。

#include<stdio.h>
int main()
{
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char arr[20];
	fgets(arr, 5, pf);
	printf("%s", arr);
	fclose(pf);
	pf == NULL;
	return 0;
}

 

 如图,只读取了4个字母。

格式化输出函数 fprintf

int fprintf ( FILE * stream, const char * format, ... );

fprintf是将数据写入文件的函数,我们直接看代码。 

#include<stdio.h>
struct S
{
	int n;
	float f;
	char arr[20];
};
int main()
{
	struct S s = { 111,5.55f,"chendadachen" };
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fprintf(pf, "%d %f %s", s.n, s.f, s.arr);
	fclose(pf);
	pf == NULL;
	return 0;
}

运行结果如下,结构体的数据成功输入。 

格式化输入函数 fscanf

我们这次将结构体置空,从上面的文件中读取信息然后打印到屏幕上。

#include<stdio.h>
struct S
{
	int n;
	float f;
	char arr[20];
};
int main()
{
	struct S s = { 0 };
	FILE *pf=fopen("C:\\Users\\HP\\Desktop\\你快乐吗.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fscanf(pf, "%d %f %s", &(s.n), &(s.f), s.arr);
	printf("%d %f %s\n", s.n, s.f, s.arr);
	fclose(pf);
	pf == NULL;
	return 0;
}

 数据成功读入。

 

sprintf 

把一个格式化的数据写到字符串中

int sprintf ( char * str, const char * format, ... );

sprintf 可以将一个结构体转换成字符串。

#include<stdio.h>
struct S
{
	int n;
	float f;
	char arr[20];
};
int main()
{
	struct S s = { 111,5.55f,"chendadachen"};
	char a[200] = { 0 };
	sprintf(a, "%d %f %s", s.n, s.f, s.arr);
	printf("%s\n", a);
	return 0;
}

 如图,以字符串的形式打印成功。

有没有办法还原回去呢?这就需要下面的sscanf了。

sscanf

将字符串转换成格式化的数据

#include<stdio.h>
struct S
{
	int n;
	float f;
	char arr[20];
};
int main()
{
	struct S s = { 111,5.55f,"chendadachen"};
	char a[200] = { 0 };
	sprintf(a, "%d %f %s", s.n, s.f, s.arr);
	printf("字符串的数据:%s\n", a);
	struct S tmp = { 0 };
	sscanf(a, "%d %f %s", &(tmp.n), &(tmp.f), tmp.arr);
	printf("格式化的数据:%d %f %s", tmp.n, tmp.f, tmp.arr);
	return 0;
}

 这篇博客旨在总结我自己阶段性的学习,要是能帮助到大家,那可真是三生有幸!如果觉得我写的不错的话还请点个赞和关注哦~我会持续输出编程的知识的!😘😘😘

有关写程序必会的C语言文件操作(上)附手绘图详解的更多相关文章

  1. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  2. ruby - 其他文件中的 Rake 任务 - 2

    我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

  3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  4. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  5. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

  6. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  7. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  8. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  9. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  10. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

随机推荐