简析sscanfsprintf 与freadfwrite

xiaoxiao2021-02-28  37

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 (“.2f”, $money); // 此时变数 $ formatted 值为 “123.10” $formatted = sprintf (“.2f”, $money); // 此时变数 $ formatted 值为 “00123.10” $formatted = sprintf (“%-08.2f”, $money); // 此时变数 $ formatted 值为 “123.1000” $formatted = sprintf (“%.2f%%”, 0.95 * 100); // 格式化为百分比 .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.

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; }
转载请注明原文地址: https://www.6miu.com/read-2622133.html

最新回复(0)