sscanf,sscanf_s 及其相关用法 #include<stdio.h> 定义函数 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 读入第二个整数。
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 是一个很好用的函数,利用它可以从字符串中取出整数、浮点数和字符串等等。
它的使用方法简单, 特别对于整数和浮点数来说。
但新手可能并不知道处理字符串时的一些高级用法, 这里做个简要说明吧。
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); //////////////////////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////// 可以用如下代码将字符串形式的 ip 地址转换为四个整数: char * inputIp int ip[4]; sscanf_s(inputIp, "%d.%d.%d.%d", &ip[0], &ip[1],&ip[2],&ip[3]);注意 sscanf_s,当读入的类型是整数或其它长度可以确定的类型时,不能在类型后面跟上长度,但 是对于字符串类型(char *)长度无法得知则必须在类型后面明确指出字符串的最大长度(即可以容纳 的空间)。
举例如下: int main(void ) { char tokenstring[] ="15 12 14..."; char s[81]; char c; int i; float fp; #include <stdio.h> #include <stdlib.h> // crt_sscanf_s.c // This program uses sscanf_s to read data items // from a string named tokenstring, then displays them. // Input various data from tokenstring: // max 80 character string plus NULL terminator sscanf_s( tokenstring, "%s", s, _countof(s) ); sscanf_s( tokenstring, "%c", &c,sizeof(char) ); sscanf_s( tokenstring, "%d", &i ); sscanf_s( tokenstring, "%f", &fp ); // Output the data read printf_s( "String = %s\n", s ); printf_s( "Character = %c\n", c ); printf_s( "Integer: = %d\n", i ); printf_s( "Real: = %f\n", fp ); }对于多个字符串读入的情况,代码如下: sscanf_s(inputString,"%s.%s.%s.%s", s1, s1.length, s2, s2.length, s3, s3.length, s4, s4.length);sscanf 函数非常好用,居然我以前一直不知道这个函数。
最近朋友用 VS2008写程序时用到这个函 数的安全版本 sscanf_s ,却出现异常问题,无法解析字符串不说,还会崩溃。
int sscanf_s( const char *buffer, const char *format [, argument ] ... ); 这是 MSDN 里面关于函数的定义,没有继续详细查看后面的备注,以及实例的情况下。
根本感觉不到 sscanf 与 sscanf_s 的区别。
以为仍然是像 sscanf 一样使用,以致出现奇怪问题。
Example: // crt_sscanf_s.c // This program uses sscanf_s to read data items // from a string named tokenstring, then displays them. #include <stdio.h> #include <stdlib.h> int main( void ) { char char char tokenstring[] = "15 12 14..."; s[81]; c;inti;float fp; // Input various data from tokenstring: // max 80 character string plus NULL terminator sscanf_s( tokenstring, "%s", s, _countof(s) ); sscanf_s( tokenstring, "%c", &c, sizeof(char) ); sscanf_s( tokenstring, "%d", &i ); sscanf_s( tokenstring, "%f", &fp ); // Output the data read printf_s( "String = %s\n", s ); printf_s( "Character = %c\n", c ); printf_s( "Integer: = %d\n", i ); printf_s( "Real: } 直到看完整个文档,看到这个实例,才发现原来还有猫腻!sscanf_s 取值的时候,需要在每个取值后面 指定取值的最大大小。
= %f\n", fp );在使用 VS2005编译一个程序时,出现了很多警告,说是用的函数是不安全的,应当使用安全版本, 即函数名称增加“_s”的版本。
警告内容: warning C4996: 'sscanf': This function or variable may be unsafe. Consider using sscanf_s instead.据了解,“_s”版本函数是微软后来对 c++做得扩展,用来替代原先不安全的函数,例如:printf、 scanf、strcpy、fopen 等等。
详细参考: ms-help://MS.VSCC.v80/MS.MSDN.v80/MS.VisualStudio.v80.chs/dv_vccrt/html/d9568b0 8-9514-49cd-b3dc-2454ded195a3.htm 原来安全版本的函数,对参数和缓冲边界做了检查,增加了返回值和抛出异常。
这样增加了函数的安 全性,减少了出错的几率。
同时这也意味着在使用这些函数时,有时你不得不输入更多的关于缓冲区大小的参数,多敲几下键盘 能换来更少的麻烦,值得! 下面总结了 sscanf 的以及 sscanf_s 的常用方法,也体现了“_s”版本函数与原函数的特别之处:1、sscanf 和 scanf 的不同是输入来源,前者是一个字符串,后者则是标准输入设备 2、sscanf 的使用,以解析时间字符串为例,将字符串“2009-01-02_11:12:13”解析为整型年月日 时分秒 //定义 char cc; tm tm_temp={0}; string stime("2009-01-02_11:12:13"); //(1) 必须严格按照分隔符形式匹配填写,若遇到不匹配项则终止解析sscanf(stime.c_str(), "%4d-%2d-%2d_%2d:%2d:%2d", &tm_temp.tm_year, &tm_temp.tm_mon, &tm_temp.tm_mday, &tm_temp.tm_hour, &tm_temp.tm_min, &tm_temp.tm_sec );//(2) 可 以 不 按 照 分 割 符 号 形 式 填 写 , 字 符 数 必 须 一 致 , 例 如 可 以 正 确 解 析 “2009/01/02_11:12:13”sscanf(stime.c_str(), "%4d%c%2d%c%2d%c%2d%c%2d%c%2d", &tm_temp.tm_year, &cc, &tm_temp.tm_mon, &cc, &tm_temp.tm_mday, &cc, &tm_temp.tm_hour, &cc,&tm_temp.tm_min, &cc, &tm_temp.tm_sec );//(3) 可以不按照分割符号形式填写,字符数必须一致,同上,%1s 可以等同于%csscanf(stime.c_str(), "%4d%1s%2d%1s%2d%1s%2d%1s%2d%1s%2d", &tm_temp.tm_year, &cc, &tm_temp.tm_mon, &cc, &tm_temp.tm_mday, &cc, &tm_temp.tm_hour, &cc, &tm_temp.tm_min, &cc, &tm_temp.tm_sec ); //(4) 可 以 不 按 照 分 割 符 形 式 和 数 量 填 写 , 类 型 必 须 一 致 , 例 如 可 以 正 确 解 析 “2009/01/02___11:12:13” //这里使用了 sscanf 的正则表达式,与通用的正则表示类似但不完全相同,%*c 表示忽略连续多个 字符sscanf(stime.c_str(), "%4d%*c%2d%*c%2d%*c%2d%*c%2d%*c%2d", &tm_temp.tm_year, &tm_temp.tm_mon, &tm_temp.tm_mday, &tm_temp.tm_hour, &tm_temp.tm_min, &tm_temp.tm_sec );3、sscanf_s 的使用//定义 char cc[2]; tm tm_temp={0}; string stime("2009-01-02_11:12:13"); //(1) 与 sscanf 第一种方法相同,可以使用"%4d-%2d-%2d_%2d:%2d:%2d"格式匹配解析sscanf_s(stime.c_str(), "%4d-%2d-%2d_%2d:%2d:%2d", &tm_temp.tm_year, &tm_temp.tm_mon, &tm_temp.tm_mday, &tm_temp.tm_hour, &tm_temp.tm_min, &tm_temp.tm_sec );//(2) 使用%c 格式对数据解析时,必须对相应的缓冲区增加长度参数,否则将会出错sscanf_s(stime.c_str(), "%4d%c%2d%c%2d%c%2d%c%2d%c%2d", &tm_temp.tm_year, &cc, 1, &tm_temp.tm_mon, &cc, 1, &tm_temp.tm_mday, &cc, 1, &tm_temp.tm_hour, &cc, 1, &tm_temp.tm_min, &cc, 1, &tm_temp.tm_sec );//(3) 使用%s 格式对数据解析时,缓冲长度必须大于字符串长度,否则不予解析sscanf_s(stime.c_str(), "%4d%1s%2d%1s%2d%1s%2d%1s%2d%1s%2d", &tm_temp.tm_year, &cc, 2, &tm_temp.tm_mon, &cc, 2, &tm_temp.tm_mday, &cc, 2, &tm_temp.tm_hour, &cc, 2, &tm_temp.tm_min, &cc, 2, &tm_temp.tm_sec ); //(4) 与 sscanf 一样,sscanf_s 同样支持正则表达式sscanf_s(stime.c_str(), "%4d%*c%2d%*c%2d%*c%2d%*c%2d%*c%2d", &tm_temp.tm_year, &tm_temp.tm_mon, &tm_temp.tm_mday, &tm_temp.tm_hour, &tm_temp.tm_min, &tm_temp.tm_sec );通过以上对比 sscanf 与 sscanf_s 的使用,可以看出后者对缓冲区安全有了更多的考虑,从而避免 了许多不经意的烦恼。