C语言奇技Y巧之sscanf函数用法详解

sscanf函数介绍

sscanf,C语言中的函数,读取格式化的字符串中的数据。

#include <stdio.h>
int sscanf(const char *str, const char *format, ...);

str:待解析的字符串;
format:字符串格式描述;
…:其后是一序列数目不定的指针参数,存储解析后的数据;
返回值:成功则返回解析参数数目,失败则返回-1,错误原因存于errno中。

一般用法:读取字符串、整型、浮点数

1、读取字符串,遇到空格停止,可指定最大读取字符个数,防止溢出。

    char buf[128] = {0};
    sscanf("1234#$%^&*/56ab  7c", "%20s", buf);
    printf("%s\n", buf);
 
    //sscanf 默认遇到空格停止,若要读取空格可以使用%[ ]

打印

1234#$%^&*/56ab

2.读取整型,固定长度格式读取每一段数据

    int year, month, day;
    int converted = sscanf("20191103", "%4d%2d%2d", &year, &month, &day);
    printf("converted=%d, year=%d, month=%d, day=%d\n", converted, year, month, day);

打印

converted=3, year=2019, month=11, day=3

3.读取浮点数,可以格式化保存小数点位数

    double lon, lat;
    int converted2 = sscanf("118.123456789 32.123456789", "%lf %lf", &lon, &lat);
    printf("converted2=%d, lon=%.9lf, lat=%lf\n", converted2, lon, lat);
    //%.9lf指定保存小数点9位,%lf默认保存6位,第七位四舍五入

打印

converted2=2, lon=118.123456789, lat=32.123457

sscanf 使用正则表达式

sscanf对正则表达式的支持不是特别强大,但大部分基本正则功能可以实现。

1.%[a-z] 读取在 a-z 之间的字符串,如果不在此之间则停止。

    char str[128] = "";
    //读取0-9和a-z和#字符,遇到不是这些字符时停止,E不属于所以停止读取
    sscanf("123#abc456#dEF789gh", "%31[0-9a-z#]", str);
    printf("str=%s\n", str);

打印

str=123#abc456#d

2.%[^a-z] 读取不在 a-z 之间的字符串,如果碰到a-z之间的字符则停止

    char str2[128] = "";
    //读取非a-z#字符,遇到a-z#字符时停止,这里遇到#停止读取
    sscanf("123#abc456#dEF789gh", "%31[^a-z#]", str2);
    printf("str2=%s\n", str2);

打印

str2=123

3.%*[^1-9] 前面带 * 号表示不保存变量,跳过符合条件的字符串。

    char str3[128] = "";
    //%*[^a-z#]读取非a-z#字符,遇到a-z#字符时停止,且不保存;%s继续读取后面所以字符
    sscanf("123#abc456#dEF789gh", "%*[^a-z#]%s", str3);
    printf("str3=%s\n", str3);

打印

str3=#abc456#dEF789gh

4.复合用法,同时读取多个字符串和整数

    std::string msg = "msg:123,ab c,456,D\"ef222";
    //分别读取字符串和整数,字符串包含空格和"符号
    char msg1[16];
    char msg2[16];
    int int1;
    int int2;
    //success=4,成功读取4个字段
    //%*[^:]前面的msg:不要,读取非:字符,遇到:停止,且不保存
    //%[a-z ]读取a-z的字符和空格,遇到其他字符则停止
    //%[a-zA-Z\"]读取a-z和A-Z和"字符,遇到其他字符则停止
    int success = sscanf(msg.c_str(),"%*[^:]:%d,%[a-z ],%d,%[a-zA-Z\"]",&int1, msg1, &int2, msg2);
    printf("success=%d\n",success);
    printf("msg1=%s\n", msg1);
    printf("msg2=%s\n", msg2);
    printf("int1=%d\n",int1);
    printf("int2=%d\n",int2);

打印

success=4
msg1=ab c
msg2=D"ef
int1=123
int2=456

获取 / 和 @ 之间的字符串

    char buf2[128] = {0};
    //%*[^/]读取非/字符,遇到/停止,这里读取了iios,且不保存
    //%[^@]读取非@字符,遇到@停止
    sscanf("iios/12DDWDFFabc@122", "%*[^/]/%[^@]", buf2);
    printf("buf2=%s\n", buf2);

打印

buf2=12DDWDFFabc

sscanf 返回-1

当sscanf第一个参数为空字符串,或第二个参数为空指针,sscanf 返回-1。
如果sscanf第一个参数为空指针,程序会崩溃。

    int b1;
    int ret_b = sscanf("123",NULL,&b1);
    printf("ret_b=%d,b1=%d,err=%d(%s)\n",ret_b, b1, errno, strerror(errno));

    int b2;
    ret_b = sscanf("","%d",&b2);
    printf("ret_b=%d,b2=%d,err=%d(%s)\n",ret_b, b2, errno, strerror(errno));

打印

ret_b=-1,b1=32766,err=22(无效的参数)
ret_b=-1,b2=436341776,err=0(成功)

sscanf 解析失败处理

如果sscanf参数语法格式没问题,不会返回-1,返回解析成功的字段个数,一个都没成功返回0。

当遇到解析失败, sscanf 会停止后面的解析,本例只解析成功了a1,解析a2时失败并停止读取,返回1,打印的a2和a3是未初始化的乱码。

    int a1;
    int a2;
    int a3;
    int ret_a = sscanf("123,a,456", "%d,%d,%d", &a1, &a2, &a3);
    printf("ret_a=%d,a1=%d,a2=%d,a3=%d\n",ret_a,a1,a2,a3);

打印

ret_a=1,a1=123,a2=1528212112,a3=32766

sscanf 缺点

1.不同操作系统对 sscanf 的支持不同,移植时可能出现bug。
2.复杂字符串处理容易出错,可读性差,不易调试。

  • 10
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值