Bootstrap

今日收获(C语言)

一.文件的打开

有这样一个结构体,它内部是文件信息区,文件信息区中的变化可以影响到硬盘中的数据。这个结构体的名字是FILE。我们如果想要写代码对文件进行各种操作,就需要一个指向文件信息区的指针,这个指针的类型是FILE*,通过它就可以间接对硬盘中的数据进行各种操作。

打开文件需要用到fopen函数,同时还需要确定打开方式,代码如下:

#include<stdio.h>
#include<string.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("text.txt", "r");//创建一个FILE类型的指针变量,指向打开的文件text.txt
	if (pf == NULL) //如果文件不存在,即pf为空指针
	{
		printf("%s", strerror(errno));
		return 1;
	}
	return 0;
}

其中printf("%s", strerror(errno));是用来打印错误信息的,要引用string.h和erron.h两个头文件。

还有一种打印错误信息的方法:

perror("fopen");

这样写不用引用string.h和erron.h两个头文件,结果会错误信息打印出来并且在前面加上“fopen:”。

常用的打开方式分为6种:

“r”:以“只读”的方式打开文件,只能从文件中读取数据。

“w”:以“只写”的方式打开文件,只能从文件中写入数据,并且每次打开时都会把所有数据清空。

有一点要注意:如果文件并不存在,以“w”方式打开时会自动创建新文件。

“a”:以“只写”的方式打开文件,可以在之前的数据后追加。

在它们的后面加上“b”,rb,wb,ab就是以二进制的形式读,写,追加数据。

二.文件的关闭

关闭文件要用到fclose函数,用法如下:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("text.txt", "r");//创建一个FILE类型的指针变量,指向打开的文件text.txt
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
    fclose(pf);//关闭文件
    pf=NULL;   //为防止pf成为野指针,将其赋为空指针
	return 0;
}

三.文件的常规读写

fputc和fgetc函数:

fputc函数用于将单个字符写入文件,一般是从键盘获取字符。这里可能有些让人疑惑,在以往的经验中,put应该是用于输出的,但这里似乎是“输入”。其实,可以把它理解成:从键盘中输入数据,输出到文件中,又称“写入”。用法如下:

#include<stdio.h>
#include<string.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("text.txt", "w");  //注意这里是“只写”的形式
	if (pf == NULL)
	{
		printf("%s", strerror(errno));
		return 1;
	}
	fputc('a', pf);  //把字符a写入pf指向的文件中
	fclose(pf);
	pf = NULL;
	return 0;
}

写入一串字符串也是没问题的:

char i = 0;
for (i = 'a';i <= 'z';i++)
{
	fputc(i, pf);
}

fgetc函数用于将字符从文件中读出,在以往的经验中get似乎是用来输入的,这里可以理解为把字符从文件中输出(读出)。用法如下:

#include<stdio.h>
#include<string.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("text.txt", "r");//注意这里是“只读”的形式
	if (pf == NULL)
	{
		printf("%s", strerror(errno));
		return 1;
	}
	char ch = 0;
	ch = fgetc(pf);     //从文件中读出一个字符,存到ch里
	printf("%c", ch);   //输出读到的字符
	fclose(pf);
	pf = NULL;
	return 0;
}

读出一串字符串也是没问题的:

char ch = 0;
while ((ch = fgetc(pf)) != EOF)//直到没有字符才停止
{
	printf("%c ", ch);
}

fputs和fgets函数

它们和上面的fputc,fgetc很像,只是把“c”变成了“s”,c代表单个字符,s代表字符串。所以它们分别用来向文件中写入或从文件中读出字符串。用法如下:

fputs:

#include<stdio.h>
#include<string.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("text.txt", "w");
	if (pf == NULL)
	{
		printf("%s", strerror(errno));
		return 1;
	}
	fputs("abcdef", pf);//将“abcdef”写入文件中
	fclose(pf);
	pf = NULL;
	return 0;
}

fgets和fputs不同,它有三个参数。分别是:地址(用来存从文件中读出的数据),最大读取字符个数(一般是数组的大小)和输入流。用法如下:

#include<stdio.h>
#include<string.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		printf("%s", strerror(errno));
		return 1;
	}
	char arr[20] = { 0 };
	fgets(arr, 20, pf);
	puts(arr);
	fclose(pf);
	pf = NULL;
	return 0;
}

fprintf和fscanf函数

这两个函数专门用于格式化数据处理,fprintf可以理解为向文件中打印数据,即“写入”;fscanf可以理解为从文件中输出数据,即“读出”。

用法如下:

fprintf:

#include<stdio.h>
struct stu   //创建一个结构体
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { "zhangsan",23 };//定义结构体变量并初始化
	FILE* pf = fopen("text.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fprintf(pf, "%s %d", s.name, s.age);
	fclose(pf);
	pf = NULL;
	return 0;
}

对比printf,fprintf多了一个参数:

fprintf(pf, "%s %d", s.name, s.age);

第一个参数是输出流(决定了应该把数据打印到哪里),第三个是要写入的数据,第二个参数是它们的格式。

fscanf:

把上面的代码稍微修改一下:

#include<stdio.h>
struct stu   //创建一个结构体
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { 0 };//定义结构体变量并初始化为0
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fscanf(pf, "%s %d", s.name, &s.age);//name本身是字符串数组名,不用取地址
	printf("%s %d", s.name, s.age);
	fclose(pf);
	pf = NULL;
	return 0;
}

fscanf第一个参数是输入流(决定了从哪里读取数据),第三个是用来存放读出的数据的地址,第二个是它们的格式。

fread和fwrite函数

这两个函数专门用来进行二进制的读写数据,如果以二进制的形式写入数据,就一定要以二进制的形式读出。用法如下:

fwrite:

#include<stdio.h>
struct stu
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { "zhangsan",34 };
	FILE* pf = fopen("text.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fwrite(&s, sizeof(struct stu), 1, pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

fwrite有四个参数:需要被写入的数据所在的地址,每个数据的字节大小,写入的数据的数量,输出流。(如果是结构体,就以结构体为单位)。

fread:

#include<stdio.h>
struct stu
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { 0 };
	FILE* pf = fopen("text.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fread(&s, sizeof(struct stu), 1, pf);
	printf("%s %d", s.name, s.age);
	fclose(pf);
	pf = NULL;
	return 0;
}

fread也有四个参数:用来存放读出的数据的地址,每个数据的字节大小,读出的数据的数量,输入流。

四.标准输入流和标准输出流

在上面的函数参数中出现了输入流,输出流这种陌生的概念,下面来解释一下:

标准输入流:FILE* stdin。标准输出流:FILE* stdout 。

所谓“标准”,其实就是键盘和屏幕。标准输入流是键盘(从键盘读取数据),标准输出流是屏幕(将数据输出(打印)到屏幕上)。去掉“标准”,输入流和输出流也可以是指向文件的指针,可以从文件中读取数据而非键盘,可以将数据输出(打印)到文件中而非屏幕。

还记得上面这串代码吗:

fprintf(pf, "%s %d", s.name, s.age);

如果我把它改一下:

fprintf(stdout , "%s %d", s.name, s.age);

它就不会把数据打印(写入)文件中,而是直接打印在屏幕上,因为我第一个参数用的是标准输出流。

再看这串代码:

fgets(arr, 20, pf);

这是在从文件中读取一串字符存到数组arr中。

但是如果我改一下:

fgets(arr, 20, stdin);

现在它就变成从键盘中读取一串字符存到数组arr中了,因为我把第三个参数改成了标准输入流。

实际上,与文件无关的代码也经常会用到fgets函数来读取字符串,因为它比gets函数安全。

五.文件的随机读写

正常情况下,读取文件数据总是从第一个字符开始的,但有些时候我们更想从某一个位置或末尾读取数据。这时候就要用到fseek,ftell 和 rewind 函数了。

fseek

首先搞明白3个概念:SEEK_SET(起始位置),SEEK_CUR(当前位置),SEEK_END(末尾)。

它们分别被宏定义为0(起始),1(当前位置),2(末尾)。

看下面的代码:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fseek(pf, 2, SEEK_SET);
	char ch = fgetc(pf);
	printf("%c\n", ch);
	fclose(pf);
	pf = NULL;
	return 0;
}

如果我文件里存的是“abcdefg”,那输出的结果应该是 c。

fseek一共有3个参数,第一个参数是文件指针(不要理解成"流"),第三个参数代表是从哪一个位置开始,第二个参数是偏移位置。

在上面的代码中,想象有一个“指针”,起始位置是a,往后偏移两位就指向了c,所以此时输出的就是c。

如果想让文件中的“指针”以现在的位置为起点继续偏移,只需要将SEEK_SET改成SEEK_CUR就可以了。

再看下面的代码:

这回从末尾开始偏移。

往左偏移相当于是反方向,偏移距离要用负数。

这两个结果其实是有些让人疑惑的:文件里存的是“abcdefg”,假如末尾指向的是 g,那往左偏移2个距离得到的应该是 e,但输出的居然是 f !下面用一张画明白:

第二个结果是“g”,这是因为输出 f 后“箭头”自动向后面移动了 1位。

ftell:

如果我想知道现在箭头离起始位置有多远,就要用到ftell函数了。

如果在上面代码的基础上加上

printf("%d", ftell(pf));

那输出的结果应该是7,因为此时箭头指向末尾。

rewind:

如果我突然想让箭头回到起始位置,就可以用到这个函数。用法如下:

rewind(pf);

六.sprintf和sscanf函数:

sprintf用于把格式化的数据转换成字符串,用法如下:

#include<stdio.h>
struct stu
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { "zhangsan",23 };
	char buf[100] = { 0 };
	sprintf(buf, "%s %d", s.name, s.age);//把name和age这两个数据全部转换成字符串,存到buf里
	printf("%s", buf);
	return 0;
}

虽然输出结果还是“zhangsan 23”,但这已经变成一个字符串了。

sscanf可以从字符串中转换出格式化的数据,用法如下:

#include<stdio.h>
struct stu
{
	char name[20];
	int age;
};
int main()
{
	struct stu s = { "zhangsan",23 };
	char buf[100] = { 0 };
	sprintf(buf, "%s %d", s.name, s.age);
	struct stu tmp = { 0 };//再定义一个结构体变量
	sscanf(buf, "%s %d", tmp.name, &tmp.age);
	printf("%s %d", tmp.name, tmp.age);
	return 0;
}

从字符串中转换出格式化的数据,存到tmp结构体的成员中。

;