先看下代码:
*head = pre; while(!feof(fp)) { fscanf(fp, "%s", str); sscanf(str, "%s,%s,%s,%s", pre->name, pre->shortcut_keys, pre->phone_no, pre->ip_addr); pst = (phonebook_info_t *)malloc(sizeof(phonebook_info_t)); if(NULL == pst) { RPT(RPT_ERR, "NULL pointer, err"); goto exit; } pre->next = pst; pre = pst; }
这段代码完成的任务是这样,读一个.CSV文件到系统中,然后将.CSV中的信息解析保存,.CSV文件的格式如下:
zhangsan,1,13601151245,192.165.54.63
maliu,2,15952694585,192.168.1.100
adc,?,13602512363,192.168.1.154
wangpin,1,13601151245,192.165.54.63
文件每一条信息有4个元素,元素之间用英文符","(逗号)隔开,想当然用sscanf的方式比较方便有效,所以代码如上所示:但是结果是每一条信息都存在了pre->name中,剩下pre->shortcut_keys等都是空。原因是我忽略了sscanf的用法,想法"相当然了",sscanf怎么知道第一个“,”是否应放入字符串呢?
正确的用法:
*head = pre; while(!feof(fp)) { fscanf(fp, "%s", str); sscanf(str, "%[^','],%[^','],%[^','],%[^',']", pre->name, pre->shortcut_keys, pre->phone_no, pre->ip_addr); pst = (phonebook_info_t *)malloc(sizeof(phonebook_info_t)); if(NULL == pst) { RPT(RPT_ERR, "NULL pointer, err"); goto exit; } pre->next = pst; pre = pst; }
sscanf(Version,"%d.%d.%d.%d",&bySelfFirst,&bySelfSecond,&bySelfThird, &bySelfFourth);
//从字符串中读特定字符。
Version是用来作输入的, 它里面有四个整数,分别赋值给后面四个数
int sscanf( const char *buffer, const char *format, ...); int sprintf(char *buffer,const char *format,... );
功能:类似于scanf和printf 但从字符串*buffer用于输入输出
1.sprintf用于格式化字符串
把变量打印到字符串中,从而获得数字的字符形式,这样不需要手工转换。
例如
char c[100];
int k=255;
sprintf(c,"%d",k);
//c包含"255"
2.sprintf用于进制转换
可以方便地得到变量的十六进制和八进制字符序列,再稍加处理即可得到每一位的值。
char c[100];
int k=255;
sprintf(c,"%x",k);
//c包含"ff"c[0]='f' c[1]='f'
3.sprintf用于连接字符串
方便地连接两个或者多个字符串
char buf[1024];
char a[100]="I ";
char b[100]="love ";
char c[100]="ACM."
sprintf(buf,"%s%s%s",a,b,c);
//buf 包含"Ilove ACM."
#include <stdio.h> int T[100]; int P[100]; int i; bool Input() { int p=0; char buf[1024]; i=0; while(scanf("%c",&buf[p])!=EOF) { if(buf[p]==10) { if(p==0)return 1; sscanf(buf,"%d %d",&T[i],&P[i]); i++; p=-1; } p++; } return 0; } void output() { int j; for(j=0;j<i;j++) { printf("%d %d/n",T[j],P[j]); } printf("/n"); } int main() { while(Input())output(); return 0; }
sscanf用于分析字符串
sscanf可以支持格式字符%[] 这为分析字符串提供了很大方便(其实scanf也支持%[])
先看一下%[]格式:
(1)-: 表示范围,如:%[1-9]表示只读取1-9这几个数字 %[a-z]表示只读取a-z小写字母,类似地 %[A-Z]只读取大写字母
(2)^: 表示不取,如:%[^1]表示读取除'1'以外的所有字符 %[^/]表示除/以外的所有字符
(3),: 范围可以用","相连接 如%[1-9,a-z]表示同时取1-9数字和a-z小写字母
(4)原则:从第一个在指定范围内的数字开始读取,到第一个不在范围内的数字结束%s 可以看成%[]的一个特例 %[^ ](注意^后面有一个空格!)
#include <stdio.h> #include <string.h> int main(int argc, char *argv[]) { char buf[1024],str1[100],str2[100],str3[100],str4[100],temp[100]="<default>"; int count; scanf("%d",&count); while(count--) { str1[0]='/0'; str2[0]='/0'; str3[0]='/0'; str4[0]='/0'; scanf("%s",buf); sscanf(buf,"%[^:]://%[^:,/]:%[,1-9]",str1,str2,str3,str4); sscanf(buf,"%[^:]://%[^:,/]/%[a-z,A-Z,/,~]",str1,str2,str4); if(str3[0]=='/0')strcpy(str3,temp); if(str4[0]=='/0')strcpy(str4,temp); printf("%s/n%s/n%s/n%s/n",str1,str2,str3,str4); } return 0; } ####################################################################### int main() { char str[]="\"7754321\"\r\n"; char buf[50]; memset(buf,'\0',sizeof(buf)); sscanf(&(str[0]),"\"%s\"",buf); printf("str=[%s],buf=[%s]\n",str,buf); }
输出结果:
str=["7754321"
],buf=[7754321"]
解决办法是:
int main() { char str[]="\"7754321\"\r\n"; char buf[50]; memset(buf,'\0',sizeof(buf)); sscanf(&(str[0]),"\"%[^\"]s\"",buf); printf("str=[%s],buf=[%s]\n",str,buf); }
转:
一、sscanf
sscanf() - 从一个 int sscanf(const char *buffer,const char *format,[argument ]...);
buffer 存储的数据 format 格式控制字符串 argument 选择性设定字符串 sscanf会从buffer里读进数据,依照argument的设定将数据写回。 字符串中读进与指定格式相符的数据.
说明;sscanf与scanf类似,都是用于输入的,只是后者以键盘(stdin)为输入源,前者以固定字符串为输入源。
1. 常见用法。
char buf[512] ; sscanf("123456 ", "%s", buf);//此处buf是数组名,它的意思是将123456以%s的形式存入buf中! printf("%s\n", buf); 结果为:123456 2. 取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。 sscanf("123456 ", "%4s", buf); printf("%s\n", buf); 结果为:1234 3. 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。 sscanf("123456 abcdedf", "%[^ ]", buf); printf("%s\n", buf); 结果为:123456
4、格式字符串可以有多个
int data1,data2,data3,data4;
char a[]="192.16.1.1";
sscanf(a,"%d.%d.%d.%d",&data1,&data2,&data3,&data4);
printf("data1=%d,data2=%d,data=3%d,data=4%d\n",data1,data2,data3,data4);
结果为:data1 =192,data2 =16,data3= 1,data4 =1;
5\格式字符串可以有多个,可以非常方便的分割IP地址和MAC地址等数据
int data1,data2,data3,data4;
char a[]="192:16:1:1";
sscanf(a,"%d:%d:%d:%d",&data1,&data2,&data3,&data4);
printf("data1=%d,data2=%d,data=3%d,data=4%d\n",data1,data2,data3,data4);
结果为:data1 =192,data2 =16,data3= 1,data4 =1;
6、类似字符串中的用法
用它来分隔类似这样的字符串2006:03:18:
int a, b, c; 以及2006:03:18 - 2006:04:18: char sztime1[16] = "", sztime2[16] = ""; sscanf("2006:03:18 - 2006:04:18", "%s - %s", sztime1, sztime2); 但是后来,我需要处理2006:03:18-2006:04:18 仅仅是取消了‘-’两边的空格,却打破了%s对字符串的界定。 我需要重新设计一个函数来处理这样的情况?这并不复杂,但是,为了使所有的代码都有统一的风格,我需要改动很多地方,把已有的sscanf替换成我自己的分割函数。我以为我肯定需要这样做,并伴随着对sscanf的强烈不满而入睡;一觉醒来,发现其实不必。 format-type中有%[]这样的type field。如果读取的字符串,不是以空格来分隔的话,就可以使用%[]。 %[]类似于一个正则表达式。[a-z]表示读取a-z的所有字符,[^a-z]表示读取除a-z以外的所有字符。 所以那个问题也就迎刃而解了: sscanf("2006:03:18 - 2006:04:18", "%[0-9,:] - %[0-9,:]", sztime1, sztime2);
8、给定一个字符串"hello, world",仅保留world。(注意:","之后有一空格,%s遇空格停止,加*则是忽略第一个读到的字符串)
sscanf("hello, world", "%*s%s", buf); printf("%s\n", buf); 结果为:world %*s表示第一个匹配到的%s被过滤掉,即hello被过滤了
二、sprintf函数:函数功能:把格式化的数据写入某个字符串
sprintf 是个变参函数,定义如下:
int sprintf( char *buffer, const char *format [, argument] ... ); 除了前两个参数类型固定外,后面可以接任意多个参数。而它的精华,显然就在第二个参数:
例子: char* who = "I"; char* whom = "CSDN"; sprintf(s, "%s love %s.", who, whom); //产生:"I love CSDN. " 这字符串写到s中 sprintf(s, "%10.3f", 3.1415626); //产生:" 3.142"