Chinaunix首页 | 论坛 | 博客
  • 博客访问: 14394
  • 博文数量: 17
  • 博客积分: 660
  • 博客等级: 上士
  • 技术积分: 180
  • 用 户 组: 普通用户
  • 注册时间: 2009-09-18 17:32
文章分类

全部博文(17)

文章存档

2011年(1)

2010年(15)

2009年(1)

我的朋友
最近访客

分类: C/C++

2010-05-07 15:23:22

表头文件 #include 定义函数 int sscanf (const char *str,const char * format,...); 函数说明 sscanf()会将参数str的字符串根据参数format字符串来转换并格式化数据。格式转换形式请参考scanf()。转换后的结果存于对应的参数内。 返回值 成功则返回参数数目,失败则返回-1,错误原因存于errno中。

# include <stdio.h> 

int main( ) 
{ 
      const char * s = "iios/12DDWDFF@122" ; 
      char buf[ 20] ; 

      sscanf ( s, "%*[^/]/%[^@]" , buf ) ; 
      printf ( "%s\n" , buf ) ; 

      return 0; 
}


结果为: 12DDWDFF sscanf与scanf类似,都是用于输入的,只是后者以屏幕( stdin ) 为输入源,前者以固定字符串为输入源。 函数原型: int scanf( const char *format [,argument]... ); 其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ' ' | '\t' | '\n' | 非%符号}, 注:
{a|b|c}表示a,b,c中选一,[d],表示可以有d也可以没有d。 width:宽度,一般可以忽略,用法如:

const char sourceStr[] = "hello, world"; 

char buf[10] = {0}; 

sscanf(sourceStr, "%5s", buf); //%5s,只取5个字符 

printf("%s",buf);


结果为:hello {h | l | I64 | L}:参数的size,通常h表示单字节size,I表示2字节 size,L表示4字节size(double例外),l64表示8字节size。 type :这就很多了,就是%s,%d之类。 特别的: %*[width] [{h | l | I64 | L}]type 表示满足该条件的被过滤掉,不会向目标参数中写入值。如: const char sourceStr[] = "hello, world"; //中间有空格 char buf[10] = {0}; sscanf(sourceStr, "%*s%s", buf); //%*s表示第一个匹配到的%s被过滤掉,即hello被过滤了 结果为:world 支持集合操作: %[a-z] 表示匹配a到z中任意字符,贪婪性(尽可能多的匹配) %[aB'] 匹配a、B、'中一员,贪婪性 %[^a] 匹配非a的任意字符,贪婪性 1. 常见用法。 以下是引用片段:   charstr[ 512] = { 0} ;   sscanf( "123456" , "%s" , str) ;   printf( "str=%s" , str) ;   2. 取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。 以下是引用片段:   sscanf( "123456" , "%4s" , str) ;   printf( "str=%s" , str) ; 结果1234   3. 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。 以下是引用片段:   sscanf( "123456abcdedf" , "%[^]" , str) ;   printf( "str=%s" , str) ; 结果123456abcdedf。   4. 取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。 以下是引用片段:   sscanf( "123456abcdedfBCDEF" , "%[1-9a-z]" , str) ;   printf( "str=%s" , str) ; 结果123456abcdedf。   5. 取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。 以下是引用片段:   sscanf( "123456abcdedfBCDEF" , "%[^A-Z]" , str) ;   printf( "str=%s" , str) ; 结果123456abcdedf。
搜集一些特殊用法: % [ ] 的用法:% [ ] 表示要读入一个字符集合, 如果[ 后面第一个字符是”^”,则表示反意思。 [ ] 内的字符串可以是1或更多字符组成。空字符集(% [ ] )是违反规定的,可 导致不可预知的结果。% [ ^ ] 也是违反规定的。 % [ a- z] 读取在 a- z 之间的字符串,如果不在此之前则停止,如 char s[ ] = "hello, my friend” ; // 注意: ,逗号在不 a-z之间 sscanf( s, “%[a-z]”, string ) ; // string=hello %[^a-z] 读取不在 a-z 之间的字符串,如果碰到a-z之间的字符则停止,如 char s[]=" HELLOkitty” ; // 注意: ,逗号在不 a-z之间 sscanf ( s, “% [ ^ a- z] ”, string ) ; // string=HELLO % * [ ^ = ] 前面带 * 号表示不保存变量。跳过符合条件的字符串。 char s[ ] = "notepad=1.0.0.1001" ; char szfilename [ 32] = "" ; int i = sscanf ( s, "%*[^=]" , szfilename ) ; // szfilename=NULL,因为没保存 int i = sscanf ( s, "%*[^=]=%s" , szfilename ) ; // szfilename=1.0.0.1001 % 40c 读取40个字符 % [ ^ = ] 读取字符串直到碰到’= ’号,’^’后面可以带更多字符, 如: char s[ ] = "notepad=1.0.0.1001" ; char szfilename [ 32] = "" ; int i = sscanf ( s, "%[^=]" , szfilename ) ; // szfilename=notepad 如果参数格式是:% [ ^ = : ] ,那么也可以从 notepad: 1. 0. 0. 1001读取notepad **************************************************************************************   很久以前,我以为c没有自己的split string函数,后来我发现了sscanf;一直以来,我以为sscanf只能以空格来界定字符串,现在我发现我错了。   sscanf是一个运行时函数,原形很简单:   int sscanf(   const char *buffer,   const char *format [, argument ] ...   );   它强大的功能体现在对format的支持上。   我以前用它来分隔类似这样的字符串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);





阅读(386) | 评论(0) | 转发(0) |
0

上一篇:字符设备驱动程序

下一篇:字节序详解

给主人留下些什么吧!~~