1.sscanf
函数原型 int sscanf(const char *buffer, const char *format, [ argument ] … );
- buffer 存储的数据
- 其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ’ ’ | ‘/t’ | ‘/n’ | 非%符号}
1、星号亦可用于格式中, (即 %d 和 %*s) 加了星号 () 表示跳过此数据不读入.
2、{a|b|c}表示a,b,c中选一,[d],表示可以有d也可以没有d。
3、width表示读取宽度。
4、{h | l | I64 | L}:参数的size,通常h表示单字节size,I表示2字节 size,L表示4字节size(double例外),l64表示8字节size。
5、type :这就很多了,就是%s,%d之类。
6、特别的:%*[width] [{h | l | I64 | L}]type 表示满足该条件的被过滤掉,不会向目标参数中写入值
支持集合操作:
%[a-z] 表示匹配a到z中任意字符,贪婪性(尽可能多的匹配)
%[aB’] 匹配a、B、’中一员,贪婪性
%[^a] 匹配非a的任意字符,贪婪性
1、一般用法
char buf[512] = ;
sscanf("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.、取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。
sscanf("123456abcdedfBCDEF", "%[1-9a-z]", buf);
printf("%s\n", buf);
结果为:123456abcdedf
5、 取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。
sscanf("123456abcdedfBCDEF", "%[^A-Z]", buf);
printf("%s\n", buf);
结果为:123456abcdedf
6、给定一个字符串iios/12DDWDFF@122,获取 / 和 @ 之间的字符串,先将 “iios/”过滤掉,再将非’@’的一串内容送到buf中
sscanf("iios/12DDWDFF@122", "%*[^/]/%[^@]", buf);
printf("%s\n", buf);
结果为:12DDWDFF
7、给定一个字符串”hello, world”,仅保留”world”。(注意:“,”之后有一空格)
sscanf(“hello, world”, "%*s%s", buf);
printf("%s\n", buf);
结果为:world
P.S.
%*s表示第一个匹配到的%s被过滤掉,即hello被过滤了, 如果没有空格则结果为NULL。
sscanf的功能很类似于正则表达式, 但却没有正则表达式强大,所以如果对于比较复杂的字符串处理,建议使用正则表达式。
2.sprintf
函数原型 int sprintf( char *buffer, const char *format, [ argument] … );
//把整数123 打印成一个字符串保存在s 中。
char s[10] = {0};
sprintf(s, "%d", 123); //产生"123"
- buffer:char型指针,指向将要写入的字符串的缓冲区。
- format:格式化字符串。
[argument]…:可选参数,可以是任何类型的数据。
返回写入buffer 的字符数,出错则返回-1. 如果 buffer 或 format 是空指针,且不出错而继续,函数将返回-1,并且 errno 会被设置为 EINVAL。
- sprintf 返回以format为格式argument为内容组成的结果被写入buffer 的字节数,结束字符‘\0’不计入内。即,如果“Hello”被写入空间足够大的buffer后,函数sprintf 返回5。同时buffer的内容将被改变。
sprintf格式的规格如下所示。[ ]中的部分是可选的。
%[指定参数][标识符][宽度][.精度]指示符
若想输出’%’本身时, 请使用’%%’处理。
1. 处理字符方向。负号时表示从后向前处理。
2. 填空字元。 0 的话表示空格填 0;空格是内定值,表示空格就放着。
3. 字符总宽度。为最小宽度。
4. 精确度。指在小数点后的浮点数位数。
转换字符
%% 印出百分比符号,不转换。
%c 整数转成对应的 ASCII 字元。
%d 整数转成十进位。
%f 倍精确度数字转成浮点数。
%o 整数转成八进位。
%s 整数转成字符串。
%x 整数转成小写十六进位。
%X 整数转成大写十六进位。
$$money = 123.1
$formatted = sprintf (“%06.2f”, $money); // 此时变数 $ formatted 值为 “123.10”
$formatted = sprintf (“%08.2f”, $money); // 此时变数 $ formatted 值为 “00123.10”
$formatted = sprintf (“%-08.2f”, $money); // 此时变数 $ formatted 值为 “123.1000”
$formatted = sprintf (“%.2f%%”, 0.95 * 100); // 格式化为百分比
%08.2f 解释:
%开始符
0是 “填空字元” 表示,如果长度不足时就用0来填满。
8格式化后总长度
2f小数位长度,即2位
第3行值为”00123.10” 解释:
因为2f是(2位)+小数点符号(1位)+前面123(3位)=6位,总长度为8位,故前面用[填空字元]0表示,即00123.10
第4行值为”123.1000” 解释:
-号为反向操作,然后填空字元0添加在最后面了
在将各种类型的数据构造成字符串时,sprintf 的强大功能很少会让你失望。由于sprintf 跟printf 在用法上几乎一样,只是打印的目的地不同而已,前者打印到字符串中,后者则直接在命令行上输出。这也导致sprintf 比printf 有用得多。
sprintf 的格式控制串中既然可以插入各种东西,并最终把它们“连成一串”,自然也就能够连接字符串,从而在许多场合可以替代strcat,但sprintf 能够一次连接多个字符串(自然也可以同时在它们中间插入别的内容,总之非常灵活)。比如:
char* who = "I";
char* whom = "优快云";
sprintf(s, "%s love %s.", who, whom); //产生:"I love 优快云. "
而当字符数组或字符缓存不是以”\0”结尾的话,直接连接就会出问题
char a1[] = {'A', 'B', 'C', 'D', 'E', 'F', 'G'};
char a2[] = {'H', 'I', 'J', 'K', 'L', 'M', 'N'};
//如果:
sprintf(s, "%s%s", a1, a2); //Don't do that!
//十有八九要出问题了。是否可以改成:
sprintf(s, "%7s%7s", a1, a2);
//也没好到哪儿去,正确的应该是:
sprintf(s, "%.7s%.7s", a1, a2);//产生:"ABCDEFGHIJKLMN"
/*这可以类比打印浮点数的”%m/nf”,在”%m.ns”中,m 表示占用宽度
(字符串长度不足时补空格,超出了则按照实际宽度打印),
n 才表示从相应的字符串中最多取用的字符数。
通常在打印字符串时m 没什么大用,还是点号后面的n 用的多。
自然,也可以前后都只取部分字符:*/
sprintf(s, "%.6s%.5s", a1, a2);//产生:"ABCDEFHIJKL"
较少有人注意printf/sprintf 函数的返回值,但有时它却是有用的,sprintf 返回了本次函数调用最终打印到字符缓冲区中的字符数目。也就是说每当一次sprinf 调用结束以后,你无须再调用一次strlen 便已经知道了结果字符串的长度。如:
int len=sprintf(s,"%d",i);
//对于正整数来说,len 便等于整数i 的10 进制位数。
3.fread
fread是一个函数,它从文件流中读数据,最多读取count个项,每个项size个字节,如果调用成功返回实际读取到的项个数(小于或等于count),如果不成功或读到文件末尾返回 0。
size_t fread ( void *buffer, size_t size, size_t count, FILE *stream) ;
- buffer
用于接收数据的内存地址
- size
要读的每个数据项的字节数,单位是字节 - count
要读count个数据项,每个数据项size个字节. - stream
输入流 - 返回真实读取的项数,若大于count则意味着产生了错误。另外,产生错误后,文件位置指示器是无法确定的。若其他stream或buffer为空指针,或在unicode模式中写入的字节数为奇数,此函数设置errno为EINVAL以及返回0.
- size
04.fwrite
fwrite() 是 C 语言标准库中的一个文件处理函数,功能是向指定的文件中写入若干数据块,如成功执行则返回实际写入的数据块数目。该函数以二进制形式对文件进行操作,不局限于文本文件。
size_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);
- 返回值:返回实际写入的数据块数目
- (1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;
- (2)size:要写入内容的单字节数;
- (3)count:要进行写入size字节的数据项的个数;
- (4)stream:目标文件指针;
- (5)返回实际写入的数据项个数count。
说明:写入到文件的哪里? 这个与文件的打开模式有关,如果是w+,则是从file pointer指向的地址开始写,替换掉之后的内容,文件的长度可以不变,stream的位置移动count个数;如果是a+,则从文件的末尾开始添加,文件长度加大。
#include <stdio.h>
struct mystruct
{
int i;
char cha;
};
int main(void)
{
FILE *stream;
struct mystruct s;
if ((stream = fopen("TEST.$$$", "wb")) == NULL) /* open file TEST.$$$ */
{
fprintf(stderr, "Cannot open output file.\n");
return 1;
}
s.i = 0;
s.cha = 'A';
fwrite(&s, sizeof(s), 1, stream); /* 写的struct文件*/
fclose(stream); /*关闭文件*/
return 0;
}