最近在作项目 ,用到了好几回sprint,这个函数功能甚是强大啊。是转换成字符串的利器。下面是我在网上找到的,感受比较好的总结。感谢各位网友,你们一块儿进步!
sprintf函数的用法
一、该函数包含在stdio.h的头文件中。
二、sprintf和平时咱们经常使用的printf函数的功能很类似。sprintf函数打印到字符串中,而printf函数打印输出到屏幕上。sprintf函数在咱们完成其余数据类型转换成字符串类型的操做中应用普遍。
三、sprintf函数的格式:
int sprintf( char *buffer, const char *format [, argument,...] );
除了前两个参数固定外,可选参数能够是任意个。buffer是字符数组名;format是格式化字符串(像:"%3d%6.2f%#x%o",%与#合用时,自动在十六进制数前面加上0x)。只要在printf中可使用的格式化字符串,在sprintf均可以使用。其中的格式化字符串是此函数的精华。
四、char str[20];
double f=14.309948;
sprintf(str,"%6.2f",f);
能够控制精度
五、char str[20];
int a=20984,b=48090;
sprintf(str,"%3d%6d",a,b);
str[]="20984 48090"
能够将多个数值数据链接起来。
六、char str[20];
char s1={'A','B','C'};
char s2={'T','Y','x'};
sprintf(str,"%.3s%.3s",s1,s2);
能够将多个字符串链接成字符串
%m.n在字符串的输出中,m表示宽度,字符串共占的列数;n表示实际的字符数。%m.n在浮点数中,m也表示宽度;n表示小数的位数。
七、能够动态指定,须要截取的字符数
char s1={'A','B','C'};
char s2={'T','Y','x'};
sprintf(str,"%.*s%.*s",2,s1,3,s2);
sprintf(s, "%*.*f", 10, 2, 3.1415926);
八、sprintf(s, "%p", &i);
能够打印出i的地址
上面的语句至关于
sprintf(s, "%0*x", 2 * sizeof(void *), &i);
九、sprintf的返回值是字符数组中字符的个数,即字符串的长度,不用在调用strlen(s)求字符串的长度html
STM32调用sprintf致使运行异常——越界
(2015-09-29 22:38:05)
void Grade3_Sub3_Debug(void)
{
char TEST_Buf[4];
sprintf( TEST_Buf, "M",Page1_Grade3_Sub3_IO);
LCD_P8x16Str(0,2,(uint8_t *)TEST_Buf);
}
编写了这么一段程序,在Cotex-M3下运行,发现一个其余部分的菜单只能进不能出,觉得是标志没作好,清零之类的问题,查找了许久并未找到问题,最后索性用SWD Debug一下,MDK提示说限制在32k,但我仍是尝试去全速运行,并在相应的语句设置了断点,发如今对应的菜单,进去以后,按了后退键,程序并无在断点上面停留下来,这个歌断点的语句是一个只要按了后退键就会执行的语句,为何却没有停下来?
缘由只有1个!那就是程序死在某一个地方了!为何程序会在按下按键的时候死掉,又没有调用中断?因此能够很确定的定位到系统出错的问题上!也就是Cotex-M3的系统异常中断处理。这时候只要点击“STOP”,那么程序就会停下,并定位到死循环的位置上面,HardFault_Handler中断。根据网友提供信息:
STM32出现HardFault_Handler故障的缘由主要有两个方面:
一、内存溢出或者访问越界。这个须要本身写程序的时候规范代码,遇到了须要慢慢排查。数组
二、堆栈溢出。增长堆栈的大小。安全
因此这时候就要去找内存溢出,数组越界的问题了,正常是数组大小设置和调用数组出问题,致使越界,函数
上面的程序通过修改:ui
void Grade3_Sub3_Debug(void)
{
char TEST_Buf[5];
sprintf( TEST_Buf, "M",Page1_Grade3_Sub3_IO);
LCD_P8x16Str(0,2,(uint8_t *)TEST_Buf);
}
也就是改变了数组大小,以后菜单进退自如了!!
这也是sprintf常出现的问题之一:
sprintf 是个变参函数,使用时常常出问题,并且只要出问题一般就是能致使程序崩溃的内存访spa
问错误,但好在由sprintf 误用致使的问题虽然严重,却很容易找出,无非就是那么几种状况,通3d
经常使用眼睛再把出错的代码多看几眼就看出来了。调试
sprintf_s()是sprintf()的安全版本,经过指定缓冲区长度来避免sprintf()存在的溢出风险 。在使用VS2008时若是你使用了sprintf函数,那么编译器会发出警告:使用sprintf存在风险,建议使用sprintf_s。这个安全版本的原型是:orm
int sprintf_s(char *buffer,size_t sizeOfBuffer,const char *format [,argument] ... );htm
缓冲区溢出
第一个参数的长度过短了,没的说,给个大点的地方吧。固然也多是后面的参数的问
题,建议变参对应必定要细心,而打印字符串时,尽可能使用”%.ns”的形式指定最大字符数。
忘记了第一个参数
低级得不能再低级问题,用printf 用得太惯了。//偶就常犯。:。(
变参对应出问题
一般是忘记了提供对应某个格式符的变参,致使之后的参数通通错位,检查检查吧。尤
其是对应”*”的那些参数,都提供了吗?不要把一个整数对应一个”%s”,编译器会以为你
欺她太甚了(编译器是obj 和exe 的妈妈,应该是个女的,:P)。
strftime
sprnitf 还有个不错的表妹:strftime,专门用于格式化时间字符串的,用法跟她表哥很像,也
是一大堆格式控制符,只是毕竟小姑娘家心细,她还要调用者指定缓冲区的最大长度,多是为
了在出现问题时能够推卸责任吧。这里举个例子:
time_t t = time(0);
//产生"YYYY-MM-DD hh:mm:ss"格式的字符串。
char s[32];
strftime(s, sizeof(s), "%Y-%m-%d %H:%M:%S", localtime(&t));
sprintf 在MFC 中也能找到他的知音:CString::Format,strftime 在MFC 中天然也有她的同道:
CTime::Format,这一对因为从面向对象哪里获得了赞助,用以写出的代码更觉优雅。