嵌入式系统中有不少场所须要解析字符串,如解析GPS的RMC帧,解析用户经过串口输入的监控命令等。通常的作法是先接受一帧字符数据,而后用sscanf函数提取相应字段。程序员
int sscanf( const char *buffer, const char *format [, argument ] ... );
sscanf
属于scanf
函数家族一员,从输入源——字符串中读取字符并根据format
给出的格式代码对它们进行转换,省略号表明可变长度的指针列表。函数
format
字符串参数中的格式代码能够由4部分组成:% [*] [width] [h|l|L]
格式码fetch
如sscanf("...", "%4hd", &short_val)
中%4hd
表示从字符串中提取宽度限制为4个数字的短整型。指针
如接受到用户输入的一个CLI命令,读取0x0012
地址的值:code
rcv_buf = "$READ:0x0012" // 从rcv_buf里面提取用户须要读取的值 fetch_num = sscanf(rcv_buf, "$READ:%x", &read_address);
若是执行成功,该函数返回值fetch_num
等于1
,read_address==0x12
,提取失败的话fetch_num==0
*sscanf
函数能够提取字符串%s
,单个字符%c
,整型%d
,%x
,能够指定提取的整型的宽度,指定提取的浮点数的小数位数等等。orm
使用该函数尤为要注意的是:第3部分的指针参数的类型如上述的read_address的类型必定要和第2部分format格式代码中的参数的%x相匹配,在提取多个值时,一不当心,后面提取的值会把前面的所有覆盖掉。举个例子:字符串
typedef struct { char month; char day; short year; }DATE_T; DATE_T current_date = {0}; fetch_num = sscanf("$TIME:20120228","$TIME:%4u%2u%2u", ¤t_date.year, ¤t_date.month, ¤t_date.day)
按照程序员的思路,提取出来的结果应该是:year = 2012, month=2, day = 28
,
可是实际结果倒是year=0, month = 2, day=28
原型
缘由就在于:%u
提取的month
是32位无符号整型,函数会把目的地址看成一个32位整形的地址,所以就杯具了,DATE_T
结构体占用4
字节,里面¤t_date.month
实际是指向第1个字节的地址,若是被看成指向unsigned int
的地址,把月份month=0x0000 0002
的值拷贝过去,前面提取的占用高2
字节的year
就会被覆盖掉,而后原本提取的year
的值变成了0x0000
编译器
sscanf
里的地址所在的参数类型保持和第2部分的format
格式化部分一致编译
typedef struct { unsigned int month; unsigned int day; unsigned int year; }DATE_T;
sscanf
函数避免了手工操做字符串之类重复发明轮子的行为,尤为注意该函数不会验证指针参数的类型是否对应format
格式代码中的正确类型。
- format格式不一样编译器处理不一样:ARM能够有空格
%u %d
,DSP C2000编译器必须是%u%d
- 有些老51单片机的编译器
format
里不支持超过4个数值的fetch
!- 对于
=%x
的format,"=0x30"与"=30"效果同样,自动剔除数字签名的0x