sscanf函数

sscanf函数的用法 收藏

头文件 #include

            定义函数 int sscanf (const char *str,const char * format,........);

            函数说明
            sscanf()会将参数str的字符串根据参数format字符串来转换并格式化数据。格式转换形式请参考scanf()。转换后的结果存于对应的参数内。


            返回值 成功则返回参数数目,失败则返回-1,错误原因存于errno中。 返回0表示失败    否则,表示正确格式化数据的个数    例如:sscanf(str,"%d%d%s", &i,&i2, &s);    如果三个变成都读入成功会返回3。    如果只读入了第一个整数到i则会返回1。证明无法从str读入第二个整数。 

            范例 #include
            main()
            {
            int i;
            unsigned int j;
            char input[ ]=”10 0x1b aaaaaaaa bbbbbbbb”;
            char s[5];
            sscanf(input,”%d %x %5[a-z] %*s %f”,&i,&j,s,s);
            printf(“%d %d %s ”,i,j,s);
            }

            执行 10 27 aaaaa


            sscanf(stringBuf.c_str(), "%20[^#]#%20[^ ]",......)语句中""中的内容含义为:

            “%[
            ]”符号用于声明字符串,它比“%s”更具体,可以用于设置读取的样式。例如“%[a-z]”只读取小写字母,读到其它字符就结束。注意,方括号中如果有“^”,代表一直读到某字符为止。例如:


            “%[^#]”:读取字符串,一直到出现“#”号为止。

            “%20[^#]”:读取20个字节的字符串,出现“#”号时结束。

            所以,“%20[^#]#%20[^ ]”的意义就是,

            读取两个20字节大小的字符串,第一个字符串可以用#结束,第二个字符串可以用回车符结束。

            它们的具体阐述,参见MSDN:“scanf Type Field Characters”章节,和“scanf Width
            Specification”章节。

*********************************************************************************************************************************************

大家都知道sscanf是一个很好用的函数,利用它可以从字符串中取出整数、浮点数和字符串等等。它的使用方法简单,特别对于整数和浮点数来说。但新手可能并不知道处理字符串时的一些高级用法,这里做个简要说明吧。

  1. 常见用法。

以下是引用片段:
  charstr[512]={0};
  sscanf("123456","%s",str);
  printf("str=%s
",str);

  2. 取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。

以下是引用片段:
  sscanf("123456","%4s",str);
  printf("str=%s
",str);

  3. 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。

以下是引用片段:
  sscanf("123456abcdedf","%[^]",str);
  printf("str=%s
",str);

  4. 取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。

以下是引用片段:
  sscanf("123456abcdedfBCDEF","%[1-9a-z]",str);
  printf("str=%s
",str);

  5. 取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。

以下是引用片段:
  sscanf("123456abcdedfBCDEF","%[^A-Z]",str);
  printf("str=%s",str);

*********************************************************************************************************************************************

名称: sscanf() - 从一个字符串中读进与指定格式相符的数据.

语法: int sscanf( string str, string fmt, mixed var1, mixed var2 ... );

整数 sscanf( 字符串 str, 字符串 fmt, 混合 var1, 混合 var2 ... );

用法: 以指定的格式 fmt 去解读字符串 str. fmt 中除了 %d 和 %s 以外, 亦可包含其他的字符串作为格式. 每一个 %d 或 %s 都对应一个参数, 按顺序为 var1, var2 ... %d 读入一个整数到参数中, 而 %s 读入一个字符串. * 亦可用于格式中, (即 %*d 和 %*s) 加了星号 (*) 表示跳过此数据不读入. (也就是不把此数据读入参数中) LPC 的 sscanf() 与 C 的 sscanf() 虽然相似, 但仍有不同之处. LPC 的 sscanf() 不需要 (也不可) 提供变量的内存位址给 sscanf(), 只需要给予变量的名字. 另一个不同点是, LPC 的 sscanf() 对于: sscanf(  str, "%s %s", str1, str2 ); 的语法, 将会把 str 中的第一个英文单字 (即第一个空白字符以前的内容) 读入 str1, 后面其余的内容读入 str2.

sscanf() 会返回符合格式的 %d 和 %s 总数.

以前曾经编写过这样的小程序:一个文本文件,每行是一条记录,每条记录中包含多个字段,每个字段之间以某种定界符分开,举例如下:

Notebook,IBM,ThinkPad X32,6,12000

(各字段以逗号分隔,内容依次是:物品名称,生产厂家,型号,数量,价格)

如果要对这样的一行记录进行处理,提取出各个字段,怎么做比较好呢?

我以前的做法是在一个循环中用strtok函数每次取一个字段,然后将内容保存到一个字符串数组中。这样做虽然可行,但我总感觉写出的代码有些啰嗦。

最近看到一段代码,用C的标准库函数sscanf,处理这样的数据,只需一行就可以了。我把代码整理了一下,去掉了无关的部分,核心部分如下:

float price;

int quantity;

char category[21], name[21];

char vendor[21], sku[21];

char buf[201];

fp = fopen(filename, "r");

fgets(buf, 200, fp);

sscanf(buf,

"%20[^#]#%20[^#]#%f#%i#%20[^#]#%20[^\n]",

name, sku, &price, &quantity, category, vendor);

下面简单做些解说:

%20[^#]# 最多读入20个字符,直到遇见定界符#,但不包含定界符

%f# 读入一个浮点数,直到遇见定界符#

%i# 读入一个整数,直到遇见定界符#

%20[^\n] 最多读入20个字符,忽略行尾的回车符

是不是很简洁明了呢?

#include <stdio.h>

int main()

{

char log[]="<14>2002-11-11 12:12:12 11.22.33.44 3 3 aaaa aaaaaa";

//char log[]="<1>2002-11-11 12:12:12 11.22.33.44 3 aaaa aaaaaa";

char test[]="<1111> 22";

char log2[200];

char str1[20];

char str2[20];

char str3[20];

char str4[20];

char str5[20];

char str6[20];

char str7[20];

int a1,a2,a3,a4,a5,a6;

sscanf(log,"<%d>%s %s %s %d %d %s",&a1,str2,str3,str4,&a5,&a6,str7);

printf("%d\n",a1);

printf("%s\n",str2);

printf("%s\n",str3);

printf("%s\n",str4);

printf("%d\n",a5);

printf("%d\n",a6);

printf("%s\n",str7);

sscanf(test,"<%d> %d",&a5,&a6);

printf("%d\n",a5);

printf("%d\n",a6);

sscanf(log,"<%[^>]>%[^ ] %[^ ] %[^ ] %[^ ] %[^ ] %[^$]",str1,str2,str3,str4,str5,str6,str7);

printf("%s\n",str1);

printf("%s\n",str2);

printf("%s\n",str3);

printf("%s\n",str4);

printf("%s\n",str5);

printf("%s\n",str6);

printf("%s\n",str7);

return 1;

}

const char *str = "drw-rw-rw- 1 user group 0 Oct 28 2003 -====流行音乐专用FTP=====-";

上面是源串,我要分别得到drw-rw-rw-,group,-====流行音乐专用FTP=====-字段

注意:因为这几个字段的值会变化,所以我要用格式化输入,分别存入下面的a b c中,高手帮忙!

下面这个是我没成功的尝试

char a[20];

char b[50];

char c[20];

int ret = sscanf(str, "%[^'' '']* %[''u''] %[^'' '']"

, a, b, c);

masterz(www.fruitfruit.com) 于 2005-8-6 17:43:49

sscanf可以支持regular expression?或许你要用boost,下面有一个简单的例子,虽然离你的问题还比较远

http://www.fruitfruit.com/vc/boost/boost_regex_test.cpp

newbiestar 于 2005-8-6 18:29:18

楼主一个问题好几个地方问……

int ret = sscanf(str, "%s%*s%*s%s%*s%*s%*s%*s%s", a, b, c);

这样就可以了,不要的东西都抛弃掉了

参见这个帖子里面我的回复

http://community.youkuaiyun.com/Expert/topic/4192/4192979.xml?temp=.2922632

ding8125(丁丁) 于 2005-8-6 19:19:12

可以通过字符数组输出一个字符串!~

今天看到一个奇怪的scanf。其实这只是用了正则表达式。

sscanf(user, "%127[^:]:%127[^ ]", user_name, password);

"%127[^:]:%127[^ ]",是正则表达式

用scanf或者printf,可以在%后面跟%d,%s等东西,也可以跟一个正则表达式。

这里,127表示最多可以接受127个字符,[^:]是正则表达式,表示非":",到":"结束

后面,%127[^ ],同样,其中[^ ]是正则表达式,表示非" ",到" "结束

所以,如果user是"wpc:123456"的字符串,那么经过上面的sscanf后,

user_name是wpc,而password是123456

### 使用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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值