scanf,sscanf利用format跳过干扰的空格

用了一点时间做读取配置部分的代码,希望一次记录上读取N个数据,

希望读取的格式就是一个IP地址加上端口号,希望把IP地址读取到4个短整数里面,端口号读取到另外的一个短整数。文字格式类似“192.120.1.120#8080”,但结果发现因为为了对齐,中间的空格干扰了读取。

读取的输入可能是这样“192.120.1.120   #     8080”,甚至可能是 " 192 .168 . 1 .120 # 8080 "

郁闷。google +MSDN,发现其实可以就用scanf,sscanf绕过。最后选择的format 是"%hu . %hu . %hu . %hu # %hu",请注意中间的空格。

参考文档:

http://www.cplusplus.com/reference/cstdio/scanf/   此文档写的很细致。

http://linux.die.net/man/3/sscanf

另外发现scanf 还具有比较简单的正则能力。处理很多格式化内容其实还是挺方便的。(原来土鳖都是过滤掉空格的。)

下面是自己测试自己测试的例子,还走了一下弯路,先以为用正则可以搞点,结果发现正则的写法还是不如空格过滤好用。

scanf的format字符串:

空格:也会同样去过滤输入字符串的空格,而且会持续过滤,直到不是空格为止。(空格也包括tab,回车等)

其他字符,(不包括%):会同样在输入字符里面跳过想用的字符。

%:控制输入一个字段,根据后面的长度,控制符读取数据内容。比如%d等。

%*:会跳过一个输入字段,比如sscanf("123  456","%*d %d",&data); data读取会是456;

%[0-9]:贪婪的读取0-9的字符,作为一个string读取

%[^0-9]:贪婪的读取非0-9的字符,作为一个string读取

 #include <iostream>
#include <stdio.h> using namespace std; int main()
{
int ret = ;
short ip_addr[];
unsigned short port; cout <<"============================================="<<std::endl;
port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ;
ret = sscanf("192.168.1.120#8080",
"%hu.%hu.%hu.%hu#%hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 5, IP 192 168 1 120Port 8080 读取正确
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ; ret = sscanf(" 192 .168 . 1 .120 # 8080",
"%hu.%hu.%hu.%hu#%hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 1, IP 192 0 0 0Port 0 读取错误
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; cout <<"============================================="<<std::endl; port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ;
ret = sscanf("192.168.1.120#8080",
"%hu%*[^.].%hu%*[^.].%hu%*[^.].%hu%*[^#]#%hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 1 IP 192 0 0 0Port 0 读取错误,因为%*[^.] 是要匹配一个任何一个非.的字符。但192后面就是.
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ;
ret = sscanf(" 192 .168 . 1 .120 # 8080",
"%hu%*[^.].%hu%*[^.].%hu%*[^.].%hu%*[^#]#%hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 5, IP 192 168 1 120Port 8080 读取正确
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; cout <<"============================================="<<std::endl; port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ;
ret = sscanf("192.168.1.120#8080",
"%hu . %hu . %hu . %hu # %hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 5, IP 192 168 1 120Port 8080 读取正确
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; port = ip_addr[] = ip_addr[] = ip_addr[] = ip_addr[] = ;
ret = sscanf(" 192 .168 . 1 .120 # 8080",
"%hu . %hu . %hu . %hu # %hu",
&ip_addr[],
&ip_addr[],
&ip_addr[],
&ip_addr[],
&port
);
//ret 5, IP 192 168 1 120Port 8080 读取正确,format中的空格的匹配输入字符串的任意空格,很好用
cout <<"ret " <<ret <<
" IP "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" "<<ip_addr[] <<" Port "<< port << endl; return ;
}

scanf,sscanf利用format跳过干扰的空格的更多相关文章

  1. 浅析C语言中printf(),sprintf(),scanf(),sscanf()的用法和区别

    printf语法: #include <stdio.h>int printf( const char *format, ... ); printf()函数根据format(格式)给出的格式 ...

  2. scanf/sscanf %[]格式控制串的用法(转)

    scanf/sscanf %[]格式控制串的用法 scanf中一种很少见但很有用的转换字符:[...]和[ ^...]. #include<stdio.h> int main() { ch ...

  3. 利用Format函数格式化时间和日期

    在做机房收费系统的时候,因为需要使用到日期进行查询,所以在数据表中没有使用自动添加日期的功能,而是采用了自定义的格式插入.但由于事先没有对时间转换的格式进行统一,导致后面查询时出现的问题不断. 插入时 ...

  4. 利用skipList(跳表)来实现排序(待补充)

    用于排名的数据结构 一般排序为利用堆排序(二叉树)和利用skipList(跳表)的方式 redis中SortedSet利用skipList(跳表)来实现排序,复杂度为O(logn),利用空间换时间,类 ...

  5. 利用cookies跳过登陆验证码

    前言在爬取某些网页时,登陆界面时经常遇到的一个坎,而现在大多数的网站在登陆时都会要求用户填写验证码.当然,我们可以设计一套机器学习的算法去破解验证码,然而,验证码的形式多种多样,稍微变一下(有些甚至是 ...

  6. 利用java生成带有干扰线的网页验证码图片

    package imageCreate; import java.awt.Color; import java.awt.Font; import java.awt.Graphics; import j ...

  7. Android开发之EditText利用键盘跳转到下一个输入框

    以前做项目的时候,从来没考虑过这些.这段时间公司内部用的一款APP,就出现了这个问题,在登录或者注册的时候,点击键盘的回车按钮,可以跳到下一个输入框的功能,这个属性一直也没记住,今天就把自己一直没记过 ...

  8. 利用canvas绘制带干扰线的验证码

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. PHP利用微信跳转的Code参数获取用户的openid

    //获取微信登录用户信息function getOpenID($appid,$appsecret,$code){   $url="https://api.weixin.qq.com/sns/ ...

随机推荐

  1. Linux下禁用、启用SeLinux

    一些Linux默认都是启用SeLinux的,在安装操作系统的时候我们可以选择开启或者关闭SeLinux,但是在安装完系统之后又如何开启与关闭呢? 在/etc/sysconf下有一个SeLinux文件, ...

  2. office开发心得——基于模板开发

    这几天正在写一个小程序,但用到生成word表格和Excel表格.到网上查了一些资料,发现如果生成表格模板相对比较固定即可把其制作成模板,需要设置什么格式可以直接在模板中设置,而程序仅需替换相应的内容即 ...

  3. C/C++内存存储

    #include <stdio.h> #include "string.h" #include "malloc.h" void Swap(int a ...

  4. next_permutation()函数 和 prev_permutation() 按字典序求全排列

    next_permutation功能:    求一个排序的下一个排列的函数,可以遍历全排列,要包含头文件<algorithm> 与之完全相反的函数还有prev_permutation 这个 ...

  5. uva12034Race

    递推,组合. 考虑第一名有i个人,则f[n]=sum(C(n,i)*f[n-i]),递推即可.. #include<cstdio> #include<algorithm> #i ...

  6. Vagrant工具

    Vagrant 是一款用来构建虚拟开发环境的工具,非常适合 php/python/ruby/java 这类语言开发 web 应用,“代码在我机子上运行没有问题”这种说辞将成为历史. 我们可以通过 Va ...

  7. UVA 1395 Slim Span (最小生成树,MST,kruscal)

    题意:给一个图,找一棵生成树,其满足:最大权-最小权=最小.简单图,不一定连通,权值可能全相同. 思路:点数量不大.根据kruscal每次挑选的是最小权值的边,那么苗条度一定也是最小.但是生成树有多棵 ...

  8. wave文件(*.wav)格式、PCM数据格式

    1. 音频简介 经常见到这样的描述: 44100HZ 16bit stereo 或者 22050HZ 8bit mono 等等. 44100HZ 16bit stereo: 每秒钟有 44100 次采 ...

  9. Java基础——I/O续

    目录 二进制I/O类 文件导航和I/O 二进制I/O类 FileInputStream类和FileOutputStream类 *FileOutputStream(file: File) *FileOu ...

  10. 【转载】两个Web.config中连接字符串中特殊字符解决方案

    userid =  test password = aps'"; 那么连接字符串的写法为: Provider=SQLOLEDB.1;Password="aps'"&quo ...