一、sprintf snprintf sprintf_s函数使用
众所周知,sprintf不能检查目标字符串的长度,可能造成众多安全问题,所以都会推荐使用snprintf.自从snprintf代替了sprintf,相信大家对snprintf的使用都不会少。
linux和windows下是不同的。linux下用的是snprintf();而windows下用的是_snprintf();
1、sprintf()
sprintf指的是字符串格式化命令.
函数声明为
int sprintf(char *string, char *format [,argument,…]);
参数:
str为要写入的字符串;
format为格式化字符串,与printf()函数相同;
argument为变量。
sprintf 跟printf 在用法上几乎一样,只是打印的目的地不同而已, 前者打印到字符串中, 后者则直接在命令行上输出。
1、sprintf()把整数打印到字符串中
如:
sprintf(s, “%d”, 123); //把整数123打印成一个字符串保存在s中
sprintf(s, “%8x”, 4567); //小写16进制,宽度占8个位置,右对齐
2、打印16 进制内容时,通常想要一种左边补0 的等宽格式
sprintf(s, “%08X”, 4567); //产生:“000011D7”
很简单,在表示宽度的数字前面加个0 就可以了。
3、控制浮点数打印格式
浮点数的打印和格式控制是sprintf 的又一大常用功能,浮点数使用格式符”%f”控制,默认保留小数点后6 位数字,比如:
sprintf(s, “%f”, 3.1415926); //产生"3.141593"
4、控制打印的宽度和小数位数
”%m.nf”格式,其中m 表示打印的宽度,n 表示小数点后的位数。比如:
sprintf(s, “%10.3f”, 3.1415626); //产生:" 3.142"sprintf(s, “%-10.3f”, 3.1415626); //产生:"3.142 "sprintf(s, “%.3f”, 3.1415626); //不指定总宽度,产生:“3.142”
C语言对数组进行操作时并不检测数组的长度,如果str的长度不够,sprintf()很容易造成缓冲区溢出,带来意想不到的后果,黑客经常利用这个弱点攻击看上去安全的系统:
#include <stdio.h>
main()
{char buf[10];sprintf(buf, "The length of the string is more than 10")printf("%s", buf);
}
运行结果为:”The length of the string is more than 10“,
同时系统提示程序已经停止。原因就是要写入的字符串的长度超过了buf的长度,造成缓冲区溢出。
使用snprintf()来代替sprintf()将能够很好的解决这个问题。
2、snprintf()
snprintf()函数用于将格式化的数据写入字符串
原型为: int snprintf(char *str, int n, char * format [, argument, …]);
参数: str为要写入的字符串;n为要写入的字符的最大数目,超过n会被截断;format为格式化字符串,与printf()函数相同;argument为变量。
返回值: 成功则返回参数str 字符串长度,失败则返回-1,错误原因存于errno 中。
snprintf()比sprintf()多了一个参数,能够控制要写入的字符串的长度,更加安全,只要稍加留意,不会造成缓冲区的溢出。
#include <stdio.h> int main(){char buffer[50];char* s = "runoobcom";int j = snprintf(buffer, 6, "%s\n", s);// 读取字符串并存储在 buffer 中printf("string:\n%s\ncharacter count = %d\n", buffer, j); // 输出 buffer及字符数return 0;}
输出结果
string:
abcde
character count = 10
对于返回值,需要注意的是snprintf的返回值是欲写入的字符串(即源字符串)长度,而不是实际写入的字符串度。如:
char test[8];
int ret= snprintf(test,5,"1234567890");
printf("%d|%s\n",ret,test);
#include <stdio.h>int main () {char a[16];size_t i;char path[216] = {0};i = snprintf(a, 13, "%012d", 12345); // 第 1 种情况printf("i = %lu, a = %s\n", i, a); // 输出:i = 12, a = 000000012345i = snprintf(a, 9, "%012d", 12345); // 第 2 种情况printf("i = %lu, a = %s\n", i, a); // 输出:i = 12, a = 00000001i = snprintf(path, sizeof(path), "%s/%s.%s.so","/system/lib64/hw/love", "tanghanyue", "default");printf("i = %lu, path = %s\n", i, path); // 输出:i = 43, path = /system/lib64/hw/love/tanghanyue.default.soreturn 0;
}
输出的结果为:
10|1234
3、sprintf_s
int sprintf_s(char *restrict buffer, rsize_t bufsz,
const char *restrict format, ...);
将数据格式化输出到字符串,sprintf_s()是sprintf()的安全版本,通过指定缓冲区长度来避免sprintf()存在的溢出风险。
sprintf_s原先只有windows的编译器才只支持,并不是C中的标准函数。
C11标准中加入了对该函数的支持,但是是可选的,并非强制加入。
C11中规定,如果编译器实现了__STDC_LIB_EXT1__ 宏,那么就要支持对该函数的实现。
gcc编译器只是部分的支持C11标准,本人测试在ubuntu的gcc 5.4.0版本中也没有实现__STDC_LIB_EXT1__ 。
gcc中可以用snprintf函数简单替代sprintf_s,但是注意2者在实现上是有一定的区别,不是完全相同。
int snprintf( char *restrict buffer, int bufsz,
const char *restrict format, ... );
二、scanf sscanf函数使用1、 scanf()
scanf函数是一个标准库函数,它的函数原型在头文件“stdio.h”中。与printf函数相同,C语言也允许在使用scanf函数之前不必包含stdio.h文件。
scanf函数的一般形式为:scanf(“格式控制字符串”, 地址表列);
scanf(“%d”,&a);
&是一个取地址运算符,&a是一个表达式,其功能是求变量的地址。
表示输入数据的类型,其格式符和意义如下
%d: 输入十进制整数
%o: 输入八进制整数
%x: 输入十六进制整数
%u: 输入无符号十进制整数
%f或e: 输入实型数(用小数形式或指数形式)
%c: 输入单个字符
%s: 输入字符串
scanf(“%d %*d %d”,&a,&b);
用以表示该输入项,读入后不赋予相应的变量,即跳过该输入值。
当输入为:1 2 3时,把1赋予a,2被跳过,3赋予b。
scanf(“%d%d”,&r,&c);
将接受输入 10 20,但遇到 10,20 则失败。
2、 sscanf()
sscanf()的作用: 从一个字符串中读进与指定格式相符的数据.
**定义函数 int sscanf (const char *str,const char * format,…);**函数说明:
sscanf()会将参数str的字符串根据参数format字符串来转换并格式化数据。格式转换形式请参考scanf()。转换后的结果存于对应的参数内。
一、描述
sscanf通常被用来解析并转换字符串,其格式定义灵活多变,可以实现很强大的字符串解析功能。
sscanf的原型
#include <stdio.h>
int sscanf(const char *str, const char *format, ...);
str:待解析的字符串;
format:字符串格式描述;
其后是一序列数目不定的指针参数,存储解析后的数据.
二、示例用法
1. sscanf的基本用法
整形数转换
int year, month, day;
int converted = sscanf("20191103", "%04d%02d%02d", &year, &month, &day);
printf("converted=%d, year=%d, month=%d, day=%d/n", converted, year, month, day);
输出结果:
converted=3, year=2019, month=11, day=03
"%04d%02d%02d"是用来解析字符串的格式,%表示格式转换的开始,d表示转换为一个整数,04作为d的修饰,表示这是一个长度为4位的整数,不足4位时以0补齐。
例子返回结果等于3,表示有3个数据成功转换,转换成功数目同时取决于被解析的字符串以及其转换格式,如果我们把例子中的格式改为"%04d%02d",那么sscanf将只返回2,day的数值不会被sscanf更改。
浮点数转换
double longitude, latitude;
int converted = sscanf("113.123456789 31.123456789", "%lf %lf", &longitude, &latitude);
printf("converted=%d, longitude=%.9lf, latitude=%lf/n", converted, longitude, latitude);
输出结果:
converted=2, longitude=113.123456789, latitude=31.123457
sscanf的格式字符串中,f表示这是一个浮点数,其修饰词l表示这是一个double的浮点数。
2. sscanf的高级用法
数字+字符串
char str[32] = "";
sscanf("123456abcdedf", "%31[0-9]", str);
printf("str=%s/n", str);
输出结果:
str=123456
上面的格式中,[0-9]表示这是一个仅包含0-9这几个字符的字符串,前面使用数字31修饰词表示这个字符串缓冲区的最大长度(这也是sscanf最为人诟病的地方,很容易出现缓冲区溢出错误,实际上sscanf是可以避免出现缓冲区溢出的,只要在书写任何字符串解析的格式时,注意加上其缓冲区尺寸的限制)。
char str[32] = "";
sscanf("123456abcdedf", "%31[0-9a-z]", str);
printf("str=%s/n", str);
输出结果:
str=123456abcdedf
在格式[]中增加了a-z的描述。
使用^示例:
char str[32] = "";
sscanf("123456abcdedf", "%31[^a-z]", str);
printf("str=%s/n", str);
输出结果:
str=123456
在[]中增加表示相反的意思,上面的[a-z]表示一个不包含任何a-z之间的字符串。
使用*的例子:
char str[32] = "";
int ret = sscanf("123456abcdedf", "%*[0-9]%31[a-z]", str);
printf("ret=%d, str=%s/n",ret, str);
输出结果:
ret=1, str=abcdedf
加上*修饰表示一个被忽略的数据,同时也不需要为它准备空间存放解析结果。如上面的例子中,我们就只使用了str一个参数存放%31[a-z]的解析结果,而sscanf也只返回1,表示只解析了一个数据。
掌握了[], ^, *如何使用后,我们会发现sscanf原来是一个如此强大的工具,很多我们原先认为必须使用正则表达式的地方,很可能使用sscanf就可以实现。
sscanf的具体用法:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int main()
{char str[100];//用法一:取指定长度的字符串sscanf("12345","%4s",str);printf("str=%s",str);//用法二:格式化时间int year,month,day,hour,minute,second;sscanf("2021/10/18 14:55:34","%d/%d/%d %d:%d:%d",&year,&month,&day,&hour,&minute,&second);printf("time = %d-%d-%d %d:%d:%d\n",year,month,day,hour,minute,second); //用法三:读入字符串sscanf("12345","%s",str);printf("str = %s\n",str);//用法四:%*d 和 %*s 加了星号 (*) 表示跳过此数据不读入. (也就是不把此数据读入参数中)sscanf("12345acc","%*d%s",str);printf("str = %s\n",str);sscanf("hello, world", "%*s%s", buf);printf("%s\n", buf);//用法五:取到指定字符为止的字符串。如在下例中,取遇到任意小写字母为止的字符串。sscanf("123456abcdedf","%[^a-z]",buf);printf("%s\n",buf);// %*s表示第一个匹配到的%s被过滤掉,即hello,被过滤了,如果没有空格则结果为NULL。//用法六:取到指定字符集为止的字符串。如在下例中,取遇到小写字母为止的字符串。sscanf("12345+acc121","%[^a-z]",str);printf("nstr = %s\n",str);return 0;
}
输出:
str=1234
time = 2021-10-18 14:55:34
str = 12345
str = acc
world
str = 123456
str = 12345+
本文链接:https://www.ngui.cc/article/show-862272.html