转自:http://hi.baidu.com/pursuitofacm/item/d1e5ecfe93055804ff358230
(附:关于解决VS2005烦人的warning: '×××' was declared deprecated http://hi.baidu.com/sky_space/item/5a1e9bde28ceab94260ae7c9
深入分析C中不安全的sprintf与strcpy http://www.jb51.net/article/37421.htm
一些不安全的常用的C函数 http://www.linuxdiyf.com/viewarticle.php?id=103351
留待后续研究)
相关函数 scanf,fscanf
表头文件: #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读入第二个整数。
例子:
#include<stdio.h>
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\n",i,j,s);
}
运行结果:
例二:
#include<stdio.h>
main()
{
int i;
unsigned int j;
char input[ ]="abcdefghij#0123456789#abcdefghij 0123456789#";
char s1[40];
char s2[40];
sscanf(input, "%20[^#]#%20[^ ]",s1,s2);
printf("%s\n%s\n",s1,s2);
}
sscanf(stringBuf.c_str(), "%20[^#]#%20[^ ]",......)语句中""中的内容含义为:
“%[]”符号用于声明字符串,它比“%s”更具体,可以用于设置读取的样式。例如“%[a-z]”只读取小写字母,读到其它字符就结束。注意,方括号中如果有“^”,代表一直读到某字符为止。
例如:
“%[^#]”:读取字符串,一直到出现“#”号为止。
“%20[^#]”:读取20个字节的字符串,出现“#”号时结束。
所以,“%20[^#]#%20[^ ]”的意义就是,
读取两个字符串(最多读入20个字符),第一个字符串可以用#结束,第二个字符串可以用回车符结束。
它们的具体阐述,参见MSDN:“scanf Type Field Characters”章节,和“scanf Width
Specification”章节。
大家都知道sscanf是一个很好用的函数,利用它可以从字符串中取出整数、浮点数和字符串等等。它的使用方法简单,特别对于整数和浮点数来说。但新手可能并不知道处理字符串时的一些高级用法,这里做个简要说明吧。
1. 常见用法。
以下是引用片段:
char str[512]={0};
sscanf("123456","%s",str);
printf("str=%s\n",str);
2. 取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。
以下是引用片段:
sscanf("123456","%4s",str);
printf("str=%s\n",str);
3. 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。
以下是引用片段:
sscanf("123456abc dedf","%[^ ]",str);
printf("str=%s\n",str);
4. 取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。
以下是引用片段:
sscanf("123456abcdedfBabck","%[1-9a-z]",str);
printf("str=%s\n",str);
5. 取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。
以下是引用片段:
sscanf("123456abcdedfBCDEF","%[^A-Z]",str);
printf("str=%s\n",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.
以前曾经编写过这样的小程序:一个文本文件,每行是一条记录,每条记录中包含多个字段,每个字段之间以某种定界符分开,举例如下:
Notebook#ThinkPad# 12000#6#X32#IBM
(各字段以逗号分隔,内容依次是:物品名称,生产厂家,型号,数量,价格)
如果要对这样的一行记录进行处理,提取出各个字段,怎么做比较好呢?
我以前的做法是在一个循环中用strtok函数每次取一个字段,然后将内容保存到一个字符串数组中。这样做虽然可行,但我总感觉写出的代码有些啰嗦。
最近看到一段代码,用C的标准库函数sscanf,处理这样的数据,只需一行就可以了。我把代码整理了一下,去掉了无关的部分,核心部分如下:
#include<stdio.h>
#include<iostream>
using namespace std;
main()
{
float price;
int quantity;
char category[21], name[21];
char vendor[21], sku[21];
char buf[201];
FILE *fp = fopen("f://1.txt", "r");
fgets(buf, 200, fp);
sscanf(buf,"%20[^#]#%20[^#]#%f#%i#%20[^#]#%20[^/n]",name, sku, &price, &quantity, category, vendor);
cout<<name<<"\n"<<sku<<"\n"<<price<<"\n"<<quantity<<"\n"<<category<<"\n"<<vendor<<"\n";
}
下面简单做些解说:
%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;
}