sprintf和printf的⽤法和运⾏异常故障
stm32 单⽚机调试使⽤sprintf语句调试程序技巧
最近在做项⽬,⽤到了好⼏次sprint,这个函数功能甚是强⼤啊。是转换成字符串的利器。下⾯是我在⽹上到的,感觉⽐较好的总结。感谢各位⽹友,⼤家⼀起进步!
sprintf函数的⽤法
1、该函数包含在stdio.h的头⽂件中。
2、sprintf和平时我们常⽤的printf函数的功能很相似。sprintf函数打印到字符串中,⽽printf函数打印输出到屏幕上。sprintf函数在我们完成其他数据类型转换成字符串类型的操作中应⽤⼴泛。
3、sprintf函数的格式:
int sprintf( char *buffer, const char *format [, argument,...] );
除了前两个参数固定外,可选参数可以是任意个。buffer是字符数组名;format是格式化字符串(像:"%3d%6.2f%#x%o",%与#合⽤时,⾃动在⼗六进制数前⾯加上0x)。只要在printf中可以使⽤的格式化字符串,在sprintf都可以使⽤。其中的格式化字符串是此函数的精华。4、char str[20];
double f=14.309948;
sprintf(str,"%6.2f",f);
可以控制精度
5、char str[20];
int a=20984,b=48090;
sprintf(str,"%3d%6d",a,b);
str[]="20984 48090"
可以将多个数值数据连接起来。
6、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表⽰⼩数的位数。7、可以动态指定,需要截取的字符数
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);
8、sprintf(s, "%p", &i);
可以打印出i的地址
上⾯的语句相当于
sprintf(s, "%0*x", 2 * sizeof(void *), &i);
9、sprintf的返回值是字符数组中字符的个数,即字符串的长度,不⽤在调⽤strlen(s)求字符串的长度
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故障的原因主要有两个⽅⾯:
1、内存溢出或者访问越界。这个需要⾃⼰写程序的时候规范代码,遇到了需要慢慢排查。
2、堆栈溢出。增加堆栈的⼤⼩。
所以这时候就要去内存溢出,数组越界的问题了,正常是数组⼤⼩设置和调⽤数组出问题,导致越界,
上⾯的程序经过修改:
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常出现的问题之⼀:
函数printf
sprintf 是个函数,使⽤时经常出,⽽且只要出问题通常就是能导致崩溃的内存访
问错误,但好在由sprintf 误⽤导致的问题虽然严重,却很容易出,⽆⾮就是那么⼏种情况,通
常⽤眼睛再把出错的代码多看⼏眼就看出来了。
sprintf_s()是sprintf()的安全版本,通过指定长度来避免sprintf()存在的溢出风险。在使⽤VS2008时如果你使⽤了sprintf函数,那么会发出警告:使⽤sprintf存在风险,建议使⽤sprintf_s。这个安全版本的原型是:
int sprintf_s( *buffer,size_t sizeOfBuffer,const char *format [,argument] ... );
第⼀个参数的长度太短了,没的说,给个⼤点的地⽅吧。当然也可能是后⾯的参数的问
题,建议变参对应⼀定要细⼼,⽽打印时,尽量使⽤”%.ns”的形式指定最⼤数。
忘记了第⼀个参数
低级得不能再低级问题,⽤printf ⽤得太惯了。//偶就常犯。:。(
变参对应出问题
通常是忘记了提供对应某个格式符的变参,导致以后的参数统统错位,检查检查吧。尤
其是对应”*”的那些参数,都提供了吗?不要把⼀个整数对应⼀个”%s”,编译器会觉得你
欺她太甚了(编译器是obj 和的妈妈,应该是个⼥的,:P)。
strftime
sprnitf 还有个不错的:strftime,专门⽤于时间字符串的,⽤法跟她很像,也
是⼀⼤堆格式控制符,只是毕竟⼩姑娘家⼼细,她还要调⽤者指定缓冲区的最⼤长度,可能是为
了在出现问题时可以推卸责任吧。这⾥举个例⼦:
time_t t = time(0);
//产⽣"YYYY-MM-DD :mm:"格式的字符串。
char s[32];
strftime(s, sizeof(s), "%Y-%m-%d %H:%M:%S", localtime(&t));
sprintf 在MFC 中也能到他的知⾳:CString::Format,strftime 在MFC 中⾃然也有她的同道:
CTime::Format,这⼀对由于从哪⾥得到了赞助,⽤以写出的代码更觉优雅。