1. 程式人生 > >sprintf和printf的 用法和執行異常故障

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常出現的問題之一:

sprintf 是個變參函式,使用時經常出問題,而且只要出問題通常就是能導致程式崩潰的記憶體訪

問錯誤,但好在由sprintf 誤用導致的問題雖然嚴重,卻很容易找出,無非就是那麼幾種情況,通

常用眼睛再把出錯的程式碼多看幾眼就看出來了。

sprintf_s()是sprintf()的安全版本,通過指定緩衝區長度來避免sprintf()存在的溢位風險 。在使用VS2008時如果你使用了sprintf函式,那麼編譯器會發出警告:使用sprintf存在風險,建議使用sprintf_s。這個安全版本的原型是:

int sprintf_s(char *buffer,size_t sizeOfBuffer,const char *format [,argument] ... );

緩衝區溢位

第一個引數的長度太短了,沒的說,給個大點的地方吧。當然也可能是後面的引數的問

題,建議變參對應一定要細心,而列印字串時,儘量使用”%.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,這一對由於從面向物件哪裡得到了贊助,用以寫出的程式碼更覺優雅。