sscanf sprintf详解

来源:互联网 发布:c语言字节是什么 编辑:程序博客网 时间:2024/05/16 12:15

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 包含"I love 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;
}

5.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;
}

-------------------------------------------------------------------------------------

sscanf用法几例

devicetype的值,首先原始字符串中是否包含devicetype=,如果包含此串儿则使用如下方式获得devicetype的值。

int main(int argc, char argv[])
{
       string szMsg("_Community=public&_MachineName=192.168.6.96&_Port=161&devicetype=_SnmpWin&seid=2");
        char *pPos = strstr(szMsg.c_str(), "devicetype=");
        if(pPos)
        {
            char *chDevID = new char[strlen(pPos)];
            if(chDevID)
            {
                memset(chDevID, 0, strlen(pPos));
                cout << pPos << endl;
                sscanf(pPos, "devicetype= %[^&]", chDevID);
                cout <<      "Device Type is:" << chDevID << endl;
                delete []chDevID;
            }
        }
}
例2:请自己比较两段代码运行结果有何不同。
#include <string>
#include <iostream>
using namespace std;
int main(int argc, char * argv[])
{
        string szMsg("0001A");
        unsigned long ulValue = 0;
        sscanf(szMsg.c_str(), "%4x", &ulValue);
        cout << ulValue << endl;
        return 0;
}
#include <string>
#include <iostream>
using namespace std;
int main(int argc, char * argv[])
{
        string szMsg("0001A");
        unsigned long ulValue = 0;
        sscanf(szMsg.c_str(), "%x", &ulValue);
        cout << ulValue << endl;
        return 0;
}
例3:还是让运行结果说话。
#include <string>
#include <iostream>
using namespace std;
int main(int argc, char * argv[])
{
        string szMsg("1000ABCD");
        char szID[32] = {0};
        sscanf(szMsg.c_str(), "%[0-9]", szID);
        cout << szID << endl;
        return 0;
}
--------------------------------------------------------------------------
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("123456 abcdedf", "%[^ ]", str);
        printf("str=%s/n", str);

4. 取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。
        以下是引用片段:
        sscanf("123456abcdedfBCDEF", "%[1-9a-z]", str);
        printf("str=%s/n", str);

5. 取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。
        以下是引用片段:
        sscanf("123456abcdedfBCDEF", "%[^A-Z]", str);
        printf("str=%s/n", str);