由sscanf用错引出bug,总结下sscanf用法

1、我使用sscanf犯下的错误

先看下代码:

    *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;
    }

2、总结sscanf用法

sscanf(Version,"%d.%d.%d.%d",&bySelfFirst,&bySelfSecond,&bySelfThird,  &bySelfFourth);


//从字符串中读特定字符。

Version是用来作输入的,   它里面有四个整数,分别赋值给后面四个数


sscanf和sprintf是scanf和printf家族的一对成员,用于处理和分析字符串非常强大得两个函数
头文件 stdio.h
原型
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."

4.sscanf用于处理输入
有些比较麻烦的输入输出用sscanf处理将会非常方便
如hrbeu1002 (这里给出的不是完整代码,其中输出部分只是简单地原样打印出来。)
因为scanf在使用除了%c以外的格式时都忽略换行符'/n',题中用空行来区分不同case的要求
显得难以处理,如果使用sscanf的话,可以先把输入用scanf存入一块缓冲区当中,再用sscanf读取
这样大大减少了工作量。
#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 可以看成%[]的一个特例 %[^ ](注意^后面有一个空格!)

这样使用sscanf+%[]可以轻松的分析字符串,很多字符串问题便迎刃而解了。
以hrbeu3001为例(不是完整代码,没有要求的格式)
只需2个sscanf函数,就能完成题目的要求,代码非常简洁。
#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;

  sscanf("2006:03:18", "%d:%d:%d", &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"





你可能感兴趣的:(c,编程语言)