fgets声明:

char *fgets(char *str, int n, FILE *stream)

描述

C 库函数 char *fgets(char *str, int n, FILE *stream) 从指定的流 stream 读取一行,并把它存储在 str 所指向的字符串内。当读取 (n-1) 个字符时,或者读取到换行符时,或者到达文件末尾时,它会停止,具体视情况而定。

参数

  • str -- 这是指向一个字符数组的指针,该数组存储了要读取的字符串。
  • n -- 这是要读取的最大字符数(包括最后的空字符)。通常是使用以 str 传递的数组长度。
  • stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要从中读取字符的流。

返回值

如果成功,该函数返回相同的 str 参数。如果到达文件末尾或者没有读取到任何字符,str 的内容保持不变,并返回一个空指针。

如果发生错误,返回一个空指针。

以上内容完全copy自:https://www.runoob.com/cprogramming/c-function-fgets.html

自己写的一个demo,用来读取一个文件中所有内容并打印:

#include <stdio.h>
#include <string.h>

int main()
{
	FILE *fp = NULL;
	char str[256];
	fp = fopen("xxx", "r"); // xxx是文件名
	if(!fp)
	{
		printf("open file failed!\n");
		return -1;
	}
        
        memset(str, 0, 256);
	while(NULL != fgets(str, 256, fp))
	{
		printf("%s", str);
	}

	fclose(fp);
	return 0;
}

结合memcmp函数可以实现读配置文件的功能

其中memcmp函数我的理解是可以判断从文件中读取到的一行是否包含指定的字符串

memcmp声明:

int memcmp(const void *str1, const void *str2, size_t n)

描述

C 库函数 int memcmp(const void *str1, const void *str2, size_t n)) 把存储区 str1 和存储区 str2 的前 n 个字节进行比较。

参数

  • str1 -- 指向内存块的指针。
  • str2 -- 指向内存块的指针。
  • n -- 要被比较的字节数。

返回值

  • 如果返回值 < 0,则表示 str1 小于 str2。
  • 如果返回值 > 0,则表示 str1 大于 str2。
  • 如果返回值 = 0,则表示 str1 等于 str2。

关于memcmp的内容完全copy自:C 库函数 – memcmp() | 菜鸟教程

我这边测试使用的配置文件如下:

test.Config:

NAME:ABC
AGE:15
ADDR:SHANGHAI

测试代码如下:

#include <stdio.h>
#include <string.h>

int main()
{
	FILE *fp = NULL;
	char str[256];

        char name[20];
	int  age;
	char addr[20];


	fp = fopen("test.Config", "r");
	if(!fp)
	{
		printf("open file failed!\n");
		return -1;
	}
        
        memset(str, 0, 256);
	while(NULL != fgets(str, 256, fp))
	{
		if(0 == memcmp(str, "NAME", sizeof("NAME")-1))
		{
			printf("name: %s\n", str);
		}
		else if(0 == memcmp(str, "AGE", sizeof("AGE")-1))
		{
			printf("age: %s\n", str);
		}
		else if(0 == memcmp(str, "ADDR", sizeof("ADDR")-1))
		{
			printf("addr: %s\n", str);
		}
		else
		{
		}

	}

	fclose(fp);
	return 0;
}

其中,sizeof("NAME")-1的目的是因为字符串最后还有一个"/0",不把它比较进去。

结合sscanf函数,可实现从配置文件读到字符串的功能,

仍然使用上面的配置文件test.Config,最终代码如下:

#include <stdio.h>
#include <string.h>

#define TEST_NAME_STRING "NAME"
#define TEST_AGE_STRING "AGE"
#define TEST_ADDR_STRING "ADDR"

int main()
{
	FILE *fp = NULL;
	char str[256];

        char name[20];
	char  age[10];
	char addr[20];


	fp = fopen("test.Config", "r");
	if(!fp)
	{
		printf("open file failed!\n");
		return -1;
	}
        
        memset(str, 0, 256);
	while(NULL != fgets(str, 256, fp))
	{
		if(0 == memcmp(str, "NAME", sizeof(TEST_NAME_STRING)-1))
		{
			sscanf(str, "%*[^:]:%s", name);
			printf("name = %s\n", name);
		}
		else if(0 == memcmp(str, "AGE", sizeof(TEST_AGE_STRING)-1))
		{
			sscanf(str, "%*[^:]:%s", age);
			printf("age = %s\n", age);
		}
		else if(0 == memcmp(str, "ADDR", sizeof(TEST_ADDR_STRING)-1))
		{
			sscanf(str, "%*[^:]:%s", addr);
			printf("addr = %s\n", addr);
		}
		else
		{
		}

	}

	fclose(fp);
	return 0;
}

简单说一下我对sscanf的理解吧,以下面这一行为例:

sscanf(str, "%*[^:]:%s", name);

此时str中的字符串内容为:NAME:ABC

其中%[^:]表示到第一个冒号为之的字符串,因此如果配置文件中使用了=号,那就换成%[^=]即可。多加了一个*号,%*[^:]表示忽略掉这个字符串,在这里,就是忽略掉了NAME,

%s之前那个冒号,就是代替实际的冒号,不必多说。

03-29 19:40