sscanf函数

sscanf函数用于从字符串读取格式化的输入,返回成功转换的项目数。示例包括按长度截取字符串和提取特定字符集的字符串。使用%[a-z]匹配小写字母,%[^a]匹配除a以外的字符,%*s则会忽略匹配到的字符串。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

C 库函数 int sscanf(const char *str, const char *format, ...) 从字符串读取格式化输入.

函数返回转换成功的个数;

用法直接贴代码;

一般用法:

int main()
{
	// 一般用法:	
	char buf[64] = {0};
	sscanf("abcdefg ", "%s", buf);
	printf("%s\n", buf);
	// 输出 abcdefg
    
   return(0);
}

按长度截取字符串:

注意是从左侧开始计算长度,并且当不足截取的长度将全部保留;

int main()
{
	// 按长度截取:	
	char buf[64] = {0};
	sscanf("abcdefg ", "%4s", buf);
	printf("%s\n", buf);
	// 输出 abcd
    
   return(0);
}

取仅包含指定字符集的字符串:取仅包含1到9和小写字母的字符串

注意:当字符串中遇到第一个非集合内的字符,则停止截取,如下只会输出“123456abcdedf” 大写字符“BCDEF”后的“xyz”不会放到buf中;

%[a-z] 表示匹配a到z中任意字符

%[aB'] 匹配a、B、'中一员

%[^a] 匹配非a的任意字符,并且停止读入

int main()
{
	// 取仅包含指定字符集的字符串:取仅包含1到9和小写字母的字符串
	char buf[64] = {0};
	sscanf("123456abcdedfBCDEFxyz", "%[1-9a-z]", buf);
	printf("%s\n", buf);
	// 输出 123456abcdedf

    
   return(0);
}

从字符串读取格式化输入.;

include <string.h>

int main()
{

   int day, year,icount;
   char weekday[20], month[20], dtm[100];

   strcpy( dtm, "Saturday May 27 2021" );
   icount = sscanf( dtm, "%s %s %d %d", weekday, month, &day, &year );
	// 成功转换四个,icount为4;
   printf("%s %d, %d = %s\n", month, day, year, weekday );
   // 输出:May 27 2021 = Saturday
   return(0);
}

其他用法:

 %*s表示第一个匹配到的%s被过滤掉

比如:

int main()
{

	char buf[512] = {0};
	sscanf("hello, world", "%*s%s", buf);
	printf("%s\n", buf);
	// 输出 world
	return(0);
}

 

### 使用C89标准中的`sscanf`函数解析嵌套JSON格式的字符串数据 尽管`sscanf`是一个强大的工具,用于从字符串中提取格式化数据[^1],但它的设计并不适合直接解析复杂的嵌套JSON结构。JSON格式通常包含动态键值对、数组和嵌套对象,而`sscanf`只能处理固定格式的字符串。然而,通过精心设计的格式化字符串和递归调用,可以实现对简单嵌套JSON结构的部分解析。 --- #### 限制与挑战 - JSON中的键值对可能包含转义字符(如`\n`或`\"`),这些需要额外处理。 - 嵌套结构(如对象和数组)需要递归解析。 - `sscanf`无法直接处理未知长度的键值对或数组元素。 --- #### 示例代码:使用`sscanf`解析嵌套JSON字符串 以下代码展示了如何使用`sscanf`解析一个简单的嵌套JSON字符串: ```c #include <stdio.h> #include <string.h> // 解析顶级JSON对象中的键值对 void parse_top_level(const char* json, const char* key, char* value, int max_len) { // 查找指定键的位置 const char* pos = strstr(json, key); if (pos == NULL) { value[0] = '\0'; return; } // 跳过冒号并读取值 pos = strchr(pos, ':'); if (pos == NULL) { value[0] = '\0'; return; } // 使用 sscanf 提取字符串值 sscanf(pos + 1, " \"%[^\"]\"", value); // 提取双引号内的内容 } // 解析嵌套JSON对象中的键值对 void parse_nested_object(const char* json, const char* object_key, const char* nested_key, char* value, int max_len) { // 查找嵌套对象的位置 const char* pos = strstr(json, object_key); if (pos == NULL) { value[0] = '\0'; return; } // 找到嵌套对象的内容 pos = strchr(pos, '{'); if (pos == NULL) { value[0] = '\0'; return; } // 提取嵌套对象的子字符串 const char* end = strchr(pos, '}'); if (end == NULL) { value[0] = '\0'; return; } int len = end - pos - 1; char nested_json[len + 1]; strncpy(nested_json, pos + 1, len); nested_json[len] = '\0'; // 在嵌套对象中查找键值对 parse_top_level(nested_json, nested_key, value, max_len); } int main() { const char* json = "{ \"name\": \"Tom\", \"age\": 20, \"local\": { \"country\": \"China\", \"city\": \"Beijing\" } }"; char name[50], country[50], city[50]; // 解析顶级键值对 parse_top_level(json, "\"name\"", name, sizeof(name)); printf("Name: %s\n", name); // 解析嵌套对象中的键值对 parse_nested_object(json, "\"local\"", "\"country\"", country, sizeof(country)); parse_nested_object(json, "\"local\"", "\"city\"", city, sizeof(city)); printf("Country: %s, City: %s\n", country, city); return 0; } ``` --- #### 输出结果 运行上述代码后,输出如下: ``` Name: Tom Country: China, City: Beijing ``` --- #### 详细说明 - **`parse_top_level`函数**:用于解析顶级JSON对象中的键值对。它通过`strstr`查找指定键的位置,并使用`sscanf`提取对应的值[^1]。 - **`parse_nested_object`函数**:用于解析嵌套JSON对象中的键值对。它首先定位嵌套对象的位置,然后提取其子字符串并递归调用`parse_top_level`进行解析。 - **转义字符处理**:在提取字符串值时,假设JSON中的字符串已经正确转义。如果需要进一步处理转义字符,可以编写辅助函数将`\n`、`\"`等转换为实际字符。 --- #### 局限性 - `sscanf`无法处理复杂的动态结构(如未知数量的键值对或数组元素)。 - 对于深度嵌套的JSON结构,递归调用可能导致代码复杂度增加。 - 如果JSON格式不规范,可能会导致解析失败。 --- #### 替代方案 对于更复杂的JSON解析需求,建议使用专门的库(如Jansson或cJSON)。这些库提供了更强大、灵活的功能,能够轻松处理嵌套结构和转义字符[^3]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

只管码

感谢您的打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值